std::format_kind
来自cppreference.com
在标头 <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>> |
(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
的主模板的程序非良构。
给定无 cv 限定的实现 input_range
的程序定义类型 T
,程序可以为 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 > 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 <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); // 非良构: // static_assert(std::format_kind<A> == std::range_format::disabled); int main() {}
参阅
(C++23) |
指示一个范围应该如何被格式化 (枚举) |