General topics | ||||||||||||||||
Flow control | ||||||||||||||||
Conditional execution statements | ||||||||||||||||
Iteration statements (loops) | ||||||||||||||||
Jump statements | ||||||||||||||||
Functions | ||||||||||||||||
Function declaration | ||||||||||||||||
Lambda function expression | ||||||||||||||||
inline specifier | ||||||||||||||||
Dynamic exception specifications(until C++17*) | ||||||||||||||||
noexcept specifier(C++11) | ||||||||||||||||
Exceptions | ||||||||||||||||
Namespaces | ||||||||||||||||
Types | ||||||||||||||||
Specifiers | ||||||||||||||||
| ||||||||||||||||
Storage duration specifiers | ||||||||||||||||
Initialization | ||||||||||||||||
Expressions | ||||||||||||||||
Alternative representations | ||||||||||||||||
Literals | ||||||||||||||||
Boolean -Integer -Floating-point | ||||||||||||||||
Character -String -nullptr(C++11) | ||||||||||||||||
User-defined(C++11) | ||||||||||||||||
Utilities | ||||||||||||||||
Attributes(C++11) | ||||||||||||||||
Types | ||||||||||||||||
typedef declaration | ||||||||||||||||
Type alias declaration(C++11) | ||||||||||||||||
Casts | ||||||||||||||||
Memory allocation | ||||||||||||||||
Classes | ||||||||||||||||
Class-specific function properties | ||||||||||||||||
| ||||||||||||||||
Special member functions | ||||||||||||||||
Templates | ||||||||||||||||
Miscellaneous | ||||||||||||||||
|
General | ||||
Literals | ||||
Operators | ||||
Conversions | ||||
Inside the definition of atemplate (bothclass template andfunction template), the meaning of some constructs may differ from one instantiation to another. In particular, types and expressions may depend on types of type template parameters and values of constant template parameters.
template<typename T>struct X: B<T>// “B<T>” is dependent on T{typename T::A* pa;// “T::A” is dependent on T// (see below for the meaning of this use of “typename”) void f(B<T>* pb){staticint i= B<T>::i;// “B<T>::i” is dependent on T pb->j++;// “pb->j” is dependent on T}};
Namelookup and binding are different for dependent names and non-dependent names.
Contents |
Non-dependent names are looked up and bound at the point of template definition. This binding holds even if at the point of template instantiation there is a better match:
If the meaning of a non-dependent name changes between the definition context and the point of instantiation of a specialization of the template, the program is ill-formed, no diagnostic required. This is possible in the following situations:
| (since C++17) |
Binding of dependent names is postponed until lookup takes place.
Thelookup of a dependent name used in a template is postponed until the template arguments are known, at which time
(in other words, adding a new function declaration after template definition does not make it visible, except via ADL).
The purpose of this rule is to help guard against violations of theODR for template instantiations:
// an external librarynamespace E{template<typename T>void writeObject(const T& t){std::cout<<"Value = "<< t<<'\n';}} // translation unit 1:// Programmer 1 wants to allow E::writeObject to work with vector<int>namespace P1{std::ostream& operator<<(std::ostream& os,conststd::vector<int>& v){for(int n: v) os<< n<<' ';return os;} void doSomething(){std::vector<int> v; E::writeObject(v);// Error: will not find P1::operator<<}} // translation unit 2:// Programmer 2 wants to allow E::writeObject to work with vector<int>namespace P2{std::ostream& operator<<(std::ostream& os,conststd::vector<int>& v){for(int n: v) os<< n<<':';return os<<"[]";} void doSomethingElse(){std::vector<int> v; E::writeObject(v);// Error: will not find P2::operator<<}}
In the above example, if non-ADL lookup foroperator<<
were allowed from the instantiation context, the instantiation ofE::writeObject<vector<int>> would have two different definitions: one usingP1::operator<< and one usingP2::operator<<. Such ODR violation may not be detected by the linker, leading to one or the other being used in both instances.
To make ADL examine a user-defined namespace, eitherstd::vector should be replaced by a user-defined class or its element type should be a user-defined class:
namespace P1{// if C is a class defined in the P1 namespacestd::ostream& operator<<(std::ostream& os,conststd::vector<C>& v){for(C n: v) os<< n;return os;} void doSomething(){std::vector<C> v; E::writeObject(v);// OK: instantiates writeObject(std::vector<P1::C>)// which finds P1::operator<< via ADL}}
Note: this rule makes it impractical to overload operators for standard library types:
#include <iostream>#include <iterator>#include <utility>#include <vector> // Bad idea: operator in global namespace, but its arguments are in std::std::ostream& operator<<(std::ostream& os,std::pair<int,double> p){return os<< p.first<<','<< p.second;} int main(){typedefstd::pair<int,double> elem_t;std::vector<elem_t> v(10);std::cout<< v[0]<<'\n';// OK, ordinary lookup finds ::operator<<std::copy(v.begin(), v.end(),std::ostream_iterator<elem_t>(std::cout," "));// Error: both ordinary lookup from the point of definition of// std::ostream_iterator and ADL will only consider the std namespace,// and will find many overloads of std::operator<<, so the lookup will be done.// Overload resolution will then fail to find operator<< for elem_t// in the set found by the lookup.}
Note: limited lookup (but not binding) of dependent names also takes place at template definition time, as needed to distinguish them from non-dependent names and also to determine whether they are members of the current instantiation or members of unknown specialization. The information obtained by this lookup can be used to detect errors, see below.
The following types aredependent types :
| (since C++11) |
The result ofdecltype applied to a type-dependent expression is a unique dependent type. Two such results refer to the same type only if their expressions areequivalent. | (since C++11) |
The pack indexing specifier applied to a type-dependent constant expression is a unique dependent type. Two such pack indexing specifiers refer to the same type only if their constant expressions are equivalent. Otherwise, two such pack indexing specifiers refer to the same type only if their indices have the same value. | (since C++26) |
Note: a typedef member of a current instantiation is only dependent when the type it refers to is.
The following expressions aretype-dependent :
| (since C++11) |
| (since C++14) |
| (since C++17) |
| (since C++26) |
(since C++17) |
| (since C++26) |
The following expressions are never type-dependent because the types of these expressions cannot be:
(since C++11) |
(since C++20) |
The following expressions arevalue-dependent :
| (since C++20) |
(since C++11) |
| (since C++11) |
(since C++17) |
This section is incomplete Reason: the lede from [temp.dep], which is missing (identifier expression followed by parenthesized list... |
This section is incomplete Reason: reword to maybe make it more clear (or at least less intimidating), and while at it, applyCWG issue 591 |
Within a class template definition (including its member functions and nested classes) some names may be deduced to refer to thecurrent instantiation. This allows certain errors to be detected at the point of definition, rather than instantiation, and removes the requirement on thetypename andtemplate disambiguators for dependent names, see below.
Only the following names can refer to the current instantiation:
A template argument is equivalent to a template parameter if
template<class T>class A{ A* p1;// A is the current instantiation A<T>* p2;// A<T> is the current instantiation::A<T>* p4;// ::A<T> is the current instantiation A<T*> p3;// A<T*> is not the current instantiation class B{ B* p1;// B is the current instantiation A<T>::B* p2;// A<T>::B is the current instantiationtypename A<T*>::B* p3;// A<T*>::B is not the current instantiation};}; template<class T>class A<T*>{ A<T*>* p1;// A<T*> is the current instantiation A<T>* p2;// A<T> is not the current instantiation}; template<int I>struct B{staticconstint my_I= I;staticconstint my_I2= I+0;staticconstint my_I3= my_I;staticconstlong my_I4= I;staticconstint my_I5=(I); B<my_I>* b1;// B<my_I> is the current instantiation:// my_I has the same type as I,// and it is initialized with only I B<my_I2>* b2;// B<my_I2> is not the current instantiation:// I + 0 is not a single identifier B<my_I3>* b3;// B<my_I3> is the current instantiation:// my_I3 has the same type as I,// and it is initialized with only my_I (which is equivalent to I) B<my_I4>* b4;// B<my_I4> is not the current instantiation:// the type of my_I4 (long) is not the same as the type of I (int) B<my_I5>* b5;// B<my_I5> is not the current instantiation:// (I) is not a single identifier};
Note that a base class can be the current instantiation if a nested class derives from its enclosing class template. Base classes that are dependent types but are not the current instantiation aredependent base classes:
template<class T>struct A{typedefint M; struct B{typedefvoid M; struct C;};}; template<class T>struct A<T>::B::C: A<T>{ M m;// OK, A<T>::M};
A name is classified as a member of the current instantiation if it is
::
) names the current instantiation and lookup finds the name in the current instantiation or in its non-dependent basetemplate<class T>class A{staticconstint i=5; int n1[i];// i refers to a member of the current instantiationint n2[A::i];// A::i refers to a member of the current instantiationint n3[A<T>::i];// A<T>::i refers to a member of the current instantiation int f();}; template<class T>int A<T>::f(){return i;// i refers to a member of the current instantiation}
Members of the current instantiation may be both dependent and non-dependent.
If the lookup of a member of current instantiation gives a different result between the point of instantiation and the point of definition, the lookup is ambiguous. Note however that when a member name is used, it is not automatically converted to a class member access expression, only explicit member access expressions indicate members of current instantiation:
struct A{int m;};struct B{int m;}; template<typename T>struct C: A, T{int f(){return this->m;}// finds A::m in the template definition contextint g(){return m;}// finds A::m in the template definition context}; templateint C<B>::f();// error: finds both A::m and B::m templateint C<B>::g();// OK: transformation to class member access syntax// does not occur in the template definition context
Within a template definition, certain names are deduced to belong to anunknown specialization, in particular,
::
is a dependent type that is not a member of the current instantiationtemplate<typename T>struct Base{}; template<typename T>struct Derived: Base<T>{void f(){// Derived<T> refers to current instantiation// there is no “unknown_type” in the current instantiation// but there is a dependent base (Base<T>)// Therefore, “unknown_type” is a member of unknown specializationtypename Derived<T>::unknown_type z;}}; template<>struct Base<int>// this specialization provides it{typedefint unknown_type;};
This classification allows the following errors to be detected at the point of template definition (rather than instantiation):
template<class T>class A{typedefint type; void f(){ A<T>::type i;// OK: “type” is a member of the current instantiationtypename A<T>::other j;// Error: // “other” is not a member of the current instantiation// and it is not a member of an unknown specialization// because A<T> (which names the current instantiation),// has no dependent bases for “other” to hide in.}};
Members of unknown specialization are always dependent, and are looked up and bound at the point of instantiation as all dependent names (see above)
In a declaration or a definition of a template, including alias template, a name that is not a member of the current instantiation and is dependent on a template parameter is not considered to be a type unless the keywordtypename is used or unless it was already established as a type name, e.g. with a typedef declaration or by being used to name a base class.
#include <iostream>#include <vector> int p=1; template<typename T>void foo(conststd::vector<T>&v){// std::vector<T>::const_iterator is a dependent name,typenamestd::vector<T>::const_iterator it= v.begin(); // without “typename”, the following is parsed as multiplication// of the type-dependent data member “const_iterator”// and some variable “p”. Since there is a global “p” visible// at this point, this template definition compiles.std::vector<T>::const_iterator* p; typedeftypenamestd::vector<T>::const_iterator iter_t; iter_t* p2;// “iter_t” is a dependent name, but it is known to be a type name} template<typename T>struct S{typedefint value_t;// member of current instantiation void f(){ S<T>::value_t n{};// S<T> is dependent, but “typename” not neededstd::cout<< n<<'\n';}}; int main(){std::vector<int> v; foo(v);// template instantiation fails: there is no member variable// called “const_iterator” in the type std::vector<int> S<int>().f();}
The keywordtypename may only be used in this way before qualified names (e.g.T::x), but the names need not be dependent.
Usualqualified name lookup is used for the identifier prefixed bytypename. Unlike the case withelaborated type specifier, the lookup rules do not change despite the qualifier:
struct A// A has a nested variable X and a nested type struct X{struct X{};int X;}; struct B{struct X{};// B has a nested type struct X}; template<class T>void f(T t){typename T::X x;} void foo(){ A a; B b; f(b);// OK: instantiates f<B>, T::X refers to B::X f(a);// error: cannot instantiate f<A>:// because qualified name lookup for A::X finds the data member}
The keywordtypename can be used even outside of templates.
#include <vector> int main(){// Both OK (after resolving CWG 382)typedeftypenamestd::vector<int>::const_iterator iter_t;typenamestd::vector<int> v;}
In some contexts, only type names can validly appear. In these contexts, a dependent qualified name is assumed to name a type and notypename is required:
| (since C++20) |
Similarly, in a template definition, a dependent name that is not a member of the current instantiation is not considered to be a template name unless the disambiguation keywordtemplate is used or unless it was already established as a template name:
template<typename T>struct S{template<typename U>void foo(){}}; template<typename T>void bar(){ S<T> s; s.foo<T>();// error: < parsed as less than operator s.template foo<T>();// OK}
The keywordtemplate may only be used in this way after operators:: (scope resolution),-> (member access through pointer), and. (member access), the following are all valid examples:
As is the case withtypename, thetemplate prefix is allowed even if the name is not dependent or the use does not appear in the scope of a template.
Even if the name to the left of::
refers to a namespace, the template disambiguator is allowed:
template<typename>struct S{}; ::template S<void> q;// allowed, but unnecessary
Due to the special rules forunqualified name lookup for template names in member access expressions, when a non-dependent template name appears in a member access expression (after-> or after.), the disambiguator is unnecessary if there is aclass or alias(since C++11) template with the same name found by ordinary lookup in the context of the expression.However, if the template found by lookup in the context of the expression differs from the one found in the context of the class, the program is ill-formed(until C++11) template<int>struct A{int value;}; template<class T>void f(T t){ t.A<0>::value;// Ordinary lookup of A finds a class template.// A<0>::value names member of class A<0>// t.A < 0; // Error: “<” is treated as the start of template argument list} | (until C++23) |
The following behavior-changing defect reports were applied retroactively to previously published C++ standards.
DR | Applied to | Behavior as published | Correct behavior |
---|---|---|---|
CWG 206 | C++98 | it was unspecified at what point semantic constraints are applied when a type used in a non-dependent name is incomplete at the point at which a template is defined but is complete at the point at which an instantiation is performed | the program is ill-formed and no diagnostic is required in this case |
CWG 224 | C++98 | the definition of dependent types was based on the form of the name rather than lookup | definition revamped |
CWG 382 | C++98 | thetypename disambiguator was only allowed in template scope | also allowed outside of templates |
CWG 468 | C++98 | thetemplate disambiguator was only allowed in template scope | also allowed outside of templates |
CWG 502 | C++98 | it was unspecified whether nested enumerations are dependent | dependent as nested classes |
CWG 1047 | C++98 | typeid expressions were never value-dependent | value-dependent if the operand is type-dependent |
CWG 1160 | C++98 | it was unspecified whether a name refers to the current instantiation when a template-id matching a primary template or partial specialization appears in the definition of a member of the template | specified |
CWG 1413 | C++98 | uninitialized static data member, static member function, and address of member of a class template were not listed as value-dependent | listed |
CWG 1471 | C++98 | a nested type of a non-dependent base of the current instantiation was dependent | it is not dependent |
CWG 1850 | C++98 | the list of cases that meaning may change between the definition context and the point of instantiation was incomplete | made complete |
CWG 1929 | C++98 | it was not clear whether thetemplate disambiguator can follow a :: where the name to its left refers to a namespace | allowed |
CWG 2066 | C++98 | this was never value-dependent | it may be value-dependent |
CWG 2100 | C++98 | address of a static data member of class template was not listed as value-dependent | listed |
CWG 2109 | C++98 | type-dependent identifier expressions might not be value-dependent | they are always value-dependent |
CWG 2276 | C++98 | a function type whose exception specification is value-dependent was not a dependent type | it is |
CWG 2307 | C++98 | a parenthesized constant template parameter used as a template argument was equivalent to that template parameter | not equivalent anymore |
CWG 2457 | C++11 | a function type with function parameter pack was not a dependent type | it is |
CWG 2785 | C++20 | requires expressions might be type-dependent | they are never type-dependent |
CWG 2905 | C++11 | anoexcept expression was only value-dependent if its operand is value-dependent | it is value-dependent if its operand involves a template parameter |
CWG 2936 | C++98 | the names of local classes of templated functions were not part of the current instantiation | they are |