std::format_kind
From cppreference.com
| ヘッダー <format> で定義 |
||
| template< class R > constexpr /* 不明 */ format_kind = /* 不明 */; |
(1) | (C++23から) |
| template< ranges::input_range R > requires std::same_as<R, std::remove_cvref_t<R>> |
(2) | (C++23から) |
変数テンプレートformat_kindは、範囲Rに適したstd::range_formatを選択します。
std::format_kind<R>は次のように定義されます。
- もしstd::same_as<std::remove_cvref_t<ranges::range_reference_t<R>>, R>がtrueならば、std::format_kind<R>はstd::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ならば、std::format_kind<R>はstd::range_format::mapです。
- それ以外の場合、std::format_kind<R>はstd::range_format::setです。
- もし
- それ以外の場合、std::format_kind<R>はstd::range_format::sequenceです。
format_kindのプライマリテンプレートをインスタンス化するプログラムは、ill-formedです。
cv修飾されていないプログラム定義型Tがinput_rangeをモデルとする場合、プログラムはTに対してformat_kindを特殊化できます。そのような特殊化は定数式で使用可能であり、型はconst std::range_formatです。
[編集] 実装例
namespace detail { template< typename > constexpr bool is_pair_or_tuple_2 = false; template< typename T, typename U > constexpr bool is_pair_or_tuple_2<std::pair<T, U>> = true; template< typename T, typename U > constexpr bool is_pair_or_tuple_2<std::tuple<T, U>> = true; template < typename T > requires std::is_reference_v<T> || std::is_const_v<T> constexpr bool is_pair_or_tuple_2<T> = is_pair_or_tuple_2<std::remove_cvref_t<T>>; } template< class R > constexpr range_format format_kind = [] { static_assert(false, "instantiating a primary template is not allowed"); return range_format::disabled; }(); template< ranges::input_range R > requires std::same_as<R, std::remove_cvref_t<R>> 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 <filesystem> #include <format> #include <map> #include <set> #include <vector> 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); // ill-formed: // static_assert(std::format_kind<A> == std::range_format::disabled); int main() {}
[編集] 関連項目
| (C++23) |
範囲をどのようにフォーマットするかを指定する (enum) |