std::format_kind

来自cppreference.com
< cpp‎ | utility‎ | format
 
 
工具库
语言支持
类型支持(基本类型、RTTI)
库功能特性测试宏 (C++20)
动态内存管理
程序工具
协程支持 (C++20)
变参数函数
调试支持
(C++26)
三路比较
(C++20)
(C++20)(C++20)(C++20)
(C++20)(C++20)(C++20)
通用工具
日期和时间
函数对象
格式化库 (C++20)
(C++11)
关系运算符 (C++20 中弃用)
整数比较函数
(C++20)(C++20)(C++20)   
(C++20)
交换类型运算
(C++14)
(C++11)
(C++11)
(C++11)
(C++17)
常用词汇类型
(C++11)
(C++17)
(C++17)
(C++17)
(C++11)
(C++17)
(C++23)
初等字符串转换
(C++17)
(C++17)

 
 
在标头 <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 为范围 R 选择适当的 std::range_format

std::format_kind<R> 的定义如下:

如果要么 Ustd::pair 的特化要么 Ustd::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() {}

参阅

指示一个范围应该如何被格式化
(枚举)