Concepts (C++)
Concepts are an extension to the templates feature provided by the C++ programming language. Concepts are named Boolean predicates on template parameters, evaluated at compile time. A concept may be associated with a template (class template, function template, member function of a class template, variable template, or alias template), in which case it serves as a constraint: it limits the set of arguments that are accepted as template parameters. Originally dating back to suggestions for C++11, the original concepts specification has been revised multiple times before formally being a required part of C++20. Main usesThe main uses of concepts are:
Constraint types and usageThere are five different places in a function template signature where a constraint can be used (labeled below as C1 to C5):[1] template<C1 T>
requires C2<T>
C3 auto Fun(C4 auto param) requires C5<T>;
The constraint forms Example: equality_comparableThe following is a declaration of the concept "equality_comparable" from the // The following concept is an implementation detail used to build equality_comparable
template<typename T, typename U>
concept weakly_equality_comparable_with = requires(const remove_reference<T>& a, const remove_reference<U>& b) {
{ a == b } -> std::same_as<bool>;
{ a != b } -> std::same_as<bool>;
{ b == a } -> std::same_as<bool>;
{ b != a } -> std::same_as<bool>;
};
template<class T>
concept equality_comparable = weakly_equality_comparable_with<T, T>;
A function template constrained on this concept may be declared as follows: void f(const equality_comparable auto&); // constrained abbreviated function template declaration using a constrained placeholder type (C4 from above)
or template <equality_comparable T>
void f(const T&); // constrained function template declaration using a type constraint (C1 from above)
And may be called as usual: f(42); // OK, int satisfies equality_comparable
Compiler diagnosticsIf a programmer attempts to use a template argument that does not satisfy the requirements of the template, the compiler will generate an error. When concepts are not used, such errors are often difficult to understand because the error is not reported in the context of the call, but rather in an internal, often deeply nested, implementation context where the type was used. For example, // std::list is typically a doubly-linked list, whose iterators are not random-access
std::list<int> l = {2, 1, 3};
std::sort(l.begin(), l.end());
Typical compiler diagnostic without concepts is over 50 lines of output, beginning with a failure to compile an expression that attempts to subtract two iterators: In instantiation of 'void std::__sort(_RandomAccessIterator, _RandomAccessIterator, _Compare) [with _RandomAccessIterator = std::_List_iterator<int>; _Compare = __gnu_cxx::__ops::_Iter_less_iter]': error: no match for 'operator-' (operand types are 'std::_List_iterator<int>' and 'std::_List_iterator<int>') std::__lg(__last - __first) * 2, [..] If concepts are used, the error can be detected and reported in the context of the call: error: cannot call function 'void std::sort(_RAIter, _RAIter) [with _RAIter = std::_List_iterator<int>]' note: concept 'RandomAccessIterator()' was not satisfied Overload resolutionConcepts can be used to choose function template overloads and class template specializations based on properties of their template arguments, as an alternative to SFINAE and tag dispatching. If an argument satisfies more than one concept, the overload associated with the more constrained concept is chosen. Type deductionConcepts may be used instead of the unconstrained type deduction placeholder auto x1 = f(y); // the type of x1 is deduced to whatever f returns
Sortable auto x2 = f(y); // the type of x2 is deduced, but only compiles if it satisfies Sortable
Implementation statusConcepts TS, as specified in ISO/IEC TS 19217:2015, are implemented as an experimental feature in GCC 6.[3] C++20 concepts are fully implemented in GCC 10,[4] MSVC 19.30,[5] and Clang 10.[6] HistoryA different form of Concepts, popularly known as "C++0x Concepts", was temporarily accepted into the working paper for C++11 but was removed in 2009.[7] In addition to concepts themselves, "C++0x Concepts" included concept maps (a feature that could make it possible, for example, for the concept "Stack" to accept In contrast to this abandoned proposal, the C++20 version of Concepts is sometimes referred to as "Concepts Lite".[8] During the C++ standards committee meeting in March 2016, the evolution working group moved to merge Concepts into the mainline C++17 standard, but the motion was defeated in full committee.[9] Concepts v1 was merged into the C++20 draft.[10] "The One Range" version of Range feature that depend on concepts was also merged into C++20. See alsoNotes
References
External links
|