Difference between revisions of "cpp/utility/format/format kind"
From cppreference.com
(Links to the definition of “program-defined type”.) |
m (-`inline`: CWG2387) |
||
(2 intermediate revisions by one user not shown) | |||
Line 6: | Line 6: | ||
{{dcl|num=1|since=c++23|1= | {{dcl|num=1|since=c++23|1= | ||
template< class R > | template< class R > | ||
− | + | constexpr /* unspecified */ format_kind = /* unspecified */; | |
}} | }} | ||
{{dcl|num=2|since=c++23|1= | {{dcl|num=2|since=c++23|1= | ||
template< ranges::input_range R > | template< ranges::input_range R > | ||
requires std::same_as<R, std::remove_cvref_t<R>> | requires std::same_as<R, std::remove_cvref_t<R>> | ||
− | + | constexpr range_format format_kind<R> = /* see description */; | |
}} | }} | ||
{{dcl end}} | {{dcl end}} | ||
Line 22: | Line 22: | ||
** If {{tt|R::mapped_type}} is valid and denotes a type, let {{tt|U}} be {{c|std::remove_cvref_t<ranges::range_reference_t<R>>}}. | ** If {{tt|R::mapped_type}} is valid and denotes a type, let {{tt|U}} be {{c|std::remove_cvref_t<ranges::range_reference_t<R>>}}. | ||
:: If either {{tt|U}} is a specialization of {{lc|std::pair}} or {{tt|U}} is a specialization of {{lc|std::tuple}} and {{c|1=std::tuple_size_v<U> == 2}}, {{c|std::format_kind<R>}} is {{c|std::range_format::map}}. | :: If either {{tt|U}} is a specialization of {{lc|std::pair}} or {{tt|U}} is a specialization of {{lc|std::tuple}} and {{c|1=std::tuple_size_v<U> == 2}}, {{c|std::format_kind<R>}} is {{c|std::range_format::map}}. | ||
− | + | :* Otherwise, {{c|std::format_kind<R>}} is {{c|std::range_format::set}}. | |
* Otherwise, {{c|std::format_kind<R>}} is {{c|std::range_format::sequence}}. | * Otherwise, {{c|std::format_kind<R>}} is {{c|std::range_format::sequence}}. | ||
A program that instantiates a primary template of the {{tt|format_kind}} variable template is ill-formed. | A program that instantiates a primary template of the {{tt|format_kind}} variable template is ill-formed. | ||
− | Given a cv-unqualified {{lsd|cpp/language/type#Program-defined type}} {{tt|T}} that models {{lconcept|input_range}}, a program can | + | Given a cv-unqualified {{lsd|cpp/language/type#Program-defined type}} {{tt|T}} that models {{lconcept|input_range}}, a program can specialize {{tt|format_kind}} for {{tt|T}}. Such specializations are usable in constant expressions, and have type {{c/core|const std::range_format}}. |
===Possible implementation=== | ===Possible implementation=== | ||
Line 35: | Line 35: | ||
{ | { | ||
template< typename > | template< typename > | ||
− | + | constexpr bool is_pair_or_tuple_2 = false; | |
template< typename T, typename U > | template< typename T, typename U > | ||
− | + | constexpr bool is_pair_or_tuple_2<std::pair<T, U>> = true; | |
− | template< typename | + | template< typename T, typename U > |
− | + | constexpr bool is_pair_or_tuple_2<std::tuple<T, U>> = true; | |
template < typename T > | template < typename T > | ||
requires std::is_reference_v<T> {{!!}} std::is_const_v<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>>; | is_pair_or_tuple_2<std::remove_cvref_t<T>>; | ||
} | } | ||
template< class R > | template< class R > | ||
− | + | constexpr range_format format_kind = [] | |
{ | { | ||
static_assert(false, "instantiating a primary template is not allowed"); | static_assert(false, "instantiating a primary template is not allowed"); | ||
Line 58: | Line 58: | ||
template< ranges::input_range R > | template< ranges::input_range R > | ||
requires std::same_as<R, std::remove_cvref_t<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>) | if constexpr (std::same_as<std::remove_cvref_t<std::ranges::range_reference_t<R>>, R>) |
Latest revision as of 06:10, 24 September 2024
Defined in header <format>
|
||
template< class R > constexpr /* unspecified */ format_kind = /* unspecified */; |
(1) | (since C++23) |
template< ranges::input_range R > requires std::same_as<R, std::remove_cvref_t<R>> |
(2) | (since C++23) |
The variable template format_kind
selects an appropriate std::range_format for a range R
.
std::format_kind<R> is defined as follows:
- If std::same_as<std::remove_cvref_t<ranges::range_reference_t<R>>, R> is true, std::format_kind<R> is std::range_format::disabled.
- Otherwise, if
R::key_type
is valid and denotes a type:- If
R::mapped_type
is valid and denotes a type, letU
be std::remove_cvref_t<ranges::range_reference_t<R>>.
- If
- If either
U
is a specialization of std::pair orU
is a specialization of std::tuple and std::tuple_size_v<U> == 2, std::format_kind<R> is std::range_format::map.
- Otherwise, std::format_kind<R> is std::range_format::set.
- If either
- Otherwise, std::format_kind<R> is std::range_format::sequence.
A program that instantiates a primary template of the format_kind
variable template is ill-formed.
Given a cv-unqualified program-defined type T
that models input_range
, a program can specialize format_kind
for T
. Such specializations are usable in constant expressions, and have type const std::range_format.
[edit] Possible implementation
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; }(); |
[edit] Example
Run this code
#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() {}
[edit] See also
(C++23) |
specifies how a range should be formatted (enum) |