std::format_kind
Материал из cppreference.com
<tbody>
</tbody>
| Определено в заголовочном файле <format>
|
||
template< class R > inline constexpr /* не указано */ format_kind = /* не указано */; |
(1) | (начиная с C++23) |
template< ranges::input_range R > requires std::same_as<R, std::remove_cvref_t<R>> inline constexpr range_format format_kind<R> = /* смотрите описание */; |
(2) | (начиная с C++23) |
Шаблонная переменная format_kind выбирает подходящий std::range_format для диапазона R.
format_kind<R> определяется следующим образом:
- Если
std::same_as<std::remove_cvref_t<ranges::range_reference_t<R>>, R>равно true,format_kind<R>равноrange_format::disabled. - Иначе если
R::key_typeдопустим и обозначает тип:- Если
R::mapped_typeдопустимо и обозначает тип, пустьUбудетstd::remove_cvref_t<ranges::range_reference_t<R>>. Если либоUявляется специализацией std::pair, либоUявляется специализацией std::tuple иstd::tuple_size_v<U> == 2,format_kind<R>равноrange_format::map. - Иначе
format_kind<R>равноrange_format::set.
- Если
- Иначе
format_kind<R>равноrange_format::sequence.
Программа, которая создаёт экземпляр основного шаблона шаблонной переменной format_kind, корректна.
Предоставленная пользователем специализация format_kind разрешена при условии, что:
Rэто cv-неквалифицированный программно-определяемый тип,Rсоответствуетinput_range,- её специализация должна использоваться в константных выражениях, и
format_kind<R>имеет типconst range_format.
Возможная реализация
namespace detail {
template< typename >
inline constexpr bool is_pair_or_tuple_2 = false;
template< typename T, typename U >
inline constexpr bool is_pair_or_tuple_2<std::pair<T, U>> = true;
template< typename... Ts >
inline constexpr bool is_pair_or_tuple_2<std::tuple<Ts...>> = true;
template < typename T >
requires std::is_reference_v<T> || std::is_const_v<T>
inline constexpr bool is_pair_or_tuple_2<T> =
is_pair_or_tuple_2<std::remove_cvref_t<T>>;
}
template< class R >
inline constexpr range_format format_kind = [] {
static_assert(false, "создание экземпляра основного шаблона не разрешено");
return range_format::disabled;
}();
template< ranges::input_range R >
requires std::same_as<R, std::remove_cvref_t<R>>
inline constexpr range_format format_kind<R> = [] {
if constexpr (std::same_as<std::remove_cvref_t<std::ranges::range_reference_t<R>>, R>)
return range_format::disabled;
else if constexpr (requires { typename R::key_type; }) {
if constexpr (requires { typename R::mapped_type; } &&
detail::is_pair_or_tuple_2<std::ranges::range_reference_t<R>>)
return range_format::map;
else
return range_format::set;
} else
return range_format::sequence;
}();
|
Пример
Запустить этот код
#include <format>
#include <filesystem>
#include <vector>
#include <set>
#include <map>
struct A {};
static_assert(std::format_kind<std::vector<int>> == std::range_format::sequence);
static_assert(std::format_kind<std::map<int>> == std::range_format::map);
static_assert(std::format_kind<std::set<int>> == std::range_format::set);
static_assert(std::format_kind<std::filesystem::path> == std::range_format::disabled);
// некорректно:
// static_assert(std::format_kind<A> == std::range_format::disabled);
int main() {}
Смотрите также
(C++23) |
указывает, как должен быть отформатирован диапазон (перечисление) |