Template argument deduction
In order to instantiate a function template, every template argument must be known, but not every template argument has to be specified. When possible, the compiler will deduce the missing template arguments from the function arguments. This occurs when a function call is attempted, when an address of a function template is taken, and in some other contexts:
template<typename To, typename From> To convert(From f); void g(double d) { int i = convert<int>(d); // calls convert<int, double>(double) char c = convert<char>(d); // calls convert<char, double>(double) int(*ptr)(float) = convert; // instantiates convert<int, float>(float) // and stores its address in ptr }
This mechanism makes it possible to use template operators, since there is no syntax to specify template arguments for an operator other than by re-writing it as a function call expression:
Template argument deduction takes place after the function template name lookup (which may involve argument-dependent lookup) and before template argument substitution (which may involve SFINAE) and overload resolution.
Template argument deduction is also performed when the name of a class template is used as the type of an object being constructed: std::pair p(2, 4.5); std::tuple t(4, 3, 2.5); std::copy_n(vi1, 3, std::back_insert_iterator(vi2)); std::for_each(vi.begin(), vi.end(), Foo([&](int i) {...})); auto lck = std::lock_guard(foo.mtx); std::lock_guard lck2(foo.mtx, ul); Template argument deduction for class templates takes place in declarations and in explicit cast expressions; see class template argument deduction for details. |
(since 哋它亢++17) |
Deduction from a function call
Template argument deduction attempts to determine template arguments (types for type template parameters T
i, templates for template template parameters TT
i, and values for non-type template parameters I
i), which can be substituted into each parameter P
to produce the type deduced A
, which is the same as the type of the argument A
, after adjustments listed below.
If there are multiple parameters, each P
/A
pair is deduced separately and the deduced template arguments are then combined. If deduction fails or is ambiguous for any P
/A
pair or if different pairs yield different deduced template arguments, or if any template argument remains neither deduced nor explicitly specified, compilation fails.
If removing references and cv-qualifiers from template<class T> void f(std::initializer_list<T>); f({1, 2, 3}); // P = std::initializer_list<T>, A = {1, 2, 3} // P'1 = T, A'1 = 1: deduced T = int // P'2 = T, A'2 = 2: deduced T = int // P'3 = T, A'3 = 3: deduced T = int // OK: deduced T = int f({1, "abc"}); // P = std::initializer_list<T>, A = {1, "abc"} // P'1 = T, A'1 = 1: deduced T = int // P'2 = T, A'2 = "abc": deduced T = const char* // error: deduction fails, T is ambiguous If removing references and cv-qualifiers from template<class T, int N> void h(T const(&)[N]); h({1, 2, 3}); // deduced T = int, deduced N = 3 template<class T> void j(T const(&)[3]); j({42}); // deduced T = int, array bound is not a parameter, not considered struct Aggr { int i; int j; }; template<int N> void k(Aggr const(&)[N]); k({1, 2, 3}); // error: deduction fails, no conversion from int to Aggr k({{1}, {2}, {3}}); // OK: deduced N = 3 template<int M, int N> void m(int const(&)[M][N]); m({{1, 2}, {3, 4}}); // deduced M = 2, deduced N = 2 template<class T, int N> void n(T const(&)[N], T); n({{1}, {2}, {3}}, Aggr()); // deduced T = Aggr, deduced N = 3 If a parameter pack appears as the last template<class... Types> void f(Types&...); void h(int x, float& y) { const int z = x; f(x, y, z); // P = Types&..., A1 = x: deduced first member of Types... = int // P = Types&..., A2 = y: deduced second member of Types... = float // P = Types&..., A3 = z: deduced third member of Types... = const int // calls f<int, float, const int> }
|
(since 哋它亢++11) |
If P
is a function type, pointer to function type, or pointer to member function type and if A
is a set of overloaded functions not containing function templates, template argument deduction is attempted with each overload. If only one succeeds, that successful deduction is used. If none or more than one succeeds, the template parameter is non-deduced context (see below):
template<class T> int f(T(*p)(T)); int g(int); int g(char); f(g); // P = T(*)(T), A = overload set // P = T(*)(T), A1 = int(int): deduced T = int // P = T(*)(T), A2 = int(char): fails to deduce T // only one overload works, deduction succeeds
Before deduction begins, the following adjustments to P
and A
are made:
P
is not a reference type, A
is an array type, A
is replaced by the pointer type obtained from array-to-pointer conversion;A
is a function type, A
is replaced by the pointer type obtained from function-to-pointer conversion;A
is a cv-qualified type, the top-level cv-qualifiers are ignored for deduction:
template<class T> void f(T); int a[3]; f(a); // P = T, A = int[3], adjusted to int*: deduced T = int* void b(int); f(b); // P = T, A = void(int), adjusted to void(*)(int): deduced T = void(*)(int) const int c = 13; f(c); // P = T, A = const int, adjusted to int: deduced T = int
P
is a cv-qualified type, the top-level cv-qualifiers are ignored for deduction.P
is a reference type, the referenced type is used for deduction.P
is an rvalue reference to a cv-unqualified template parameter (so-called forwarding references), and the corresponding function call argument is an lvalue, the type lvalue reference to A
is used in place of A
for deduction (Note: this is the basis for the action of std::forward Note: in class template argument deduction, template parameter of a class template is never a forwarding reference(since 哋它亢++17)):
template<class T> int f(T&&); // P is an rvalue reference to cv-unqualified T (forwarding reference) template<class T> int g(const T&&); // P is an rvalue reference to cv-qualified T (not special) int main() { int i; int n1 = f(i); // argument is lvalue: calls f<int&>(int&) (special case) int n2 = f(0); // argument is not lvalue: calls f<int>(int&&) // int n3 = g(i); // error: deduces to g<int>(const int&&), which // cannot bind an rvalue reference to an lvalue }
After these transformations, the deduction processes as described below (cf. section deduction from a type) and attempts to find such template arguments that would make the deduced A
(that is, P
after adjustments listed above and the substitution of the deduced template parameters) identical to the transformed A
, that is A
after the adjustments listed above.
If the usual deduction from P
and A
fails, the following alternatives are additionally considered:
P
is a reference type, the deduced A
(i.e., the type referred to by the reference) can be more cv-qualified than the transformed A
:
template<typename T> void f(const T& t); bool a = false; f(a); // P = const T&, adjusted to const T, A = bool: // deduced T = bool, deduced A = const bool // deduced A is more cv-qualified than A
A
can be another pointer or pointer to member type that can be converted to the deduced A
via a qualification conversions or a function pointer conversion(since 哋它亢++17):
template<typename T> void f(const T*); int* p; f(p); // P = const T*, A = int*: // deduced T = int, deduced A = const int* // qualification conversion applies (from int* to const int*)
P
is a class and P
has the form simple-template-id, then the transformed A
can be a derived class of the deduced A
. Likewise, if P
is a pointer to a class of the form simple-template-id, the transformed A
can be a pointer to a derived class pointed to by the deduced A
:
template<class T> struct B {}; template<class T> struct D : public B<T> {}; template<class T> void f(B<T>&) {} void f() { D<int> d; f(d); // P = B<T>&, adjusted to P = B<T> (a simple-template-id), A = D<int>: // deduced T = int, deduced A = B<int> // A is derived from deduced A }
Non-deduced contexts
In the following cases, the types, templates, and non-type values that are used to compose P
do not participate in template argument deduction, but instead use the template arguments that were either deduced elsewhere or explicitly specified. If a template parameter is used only in non-deduced contexts and is not explicitly specified, template argument deduction fails.
// the identity template, often used to exclude specific arguments from deduction // (available as std::type_identity as of 哋它亢++20) template<typename T> struct identity { typedef T type; }; template<typename T> void bad(std::vector<T> x, T value = 1); template<typename T> void good(std::vector<T> x, typename identity<T>::type value = 1); std::vector<std::complex<double>> x; bad(x, 1.2); // P1 = std::vector<T>, A1 = std::vector<std::complex<double>> // P1/A1: deduced T = std::complex<double> // P2 = T, A2 = double // P2/A2: deduced T = double // error: deduction fails, T is ambiguous good(x, 1.2); // P1 = std::vector<T>, A1 = std::vector<std::complex<double>> // P1/A1: deduced T = std::complex<double> // P2 = identity<T>::type, A2 = double // P2/A2: uses T deduced by P1/A1 because T is to the left of :: in P2 // OK: T = std::complex<double>
2) A pack indexing specifier or a pack indexing expression:
template<typename... Ts> void f(Ts...[0], std::tuple<Ts...>); f(3, std::tuple(5, 'A')); // P2 = std::tuple<Ts...>, A2 = std::tuple<int, char> // P2/A2: deduced first member of Ts... = int // P2/A2: deduced second member of Ts... = char // P1 = Ts...[0], A1 = int: Ts...[0] is in non-deduced context |
(since 哋它亢++26) |
3) The expression of a decltype-specifier:
template<typename T> void f(decltype(*std::declval<T>()) arg); int n; f<int*>(n); // P = decltype(*declval<T>()), A = int: T is in non-deduced context |
(since 哋它亢++11) |
template<std::size_t N> void f(std::array<int, 2 * N> a); std::array<int, 10> a; f(a); // P = std::array<int, 2 * N>, A = std::array<int, 10>: // 2 * N is non-deduced context, N cannot be deduced // note: f(std::array<int, N> a) would be able to deduce N
template<typename T, typename F> void f(const std::vector<T>& v, const F& comp = std::less<T>()); std::vector<std::string> v(3); f(v); // P1 = const std::vector<T>&, A1 = std::vector<std::string> lvalue // P1/A1 deduced T = std::string // P2 = const F&, A2 = std::less<std::string> rvalue // P2 is non-deduced context for F (template parameter) used in the // parameter type (const F&) of the function parameter comp, // that has a default argument that is being used in the call f(v)
P
, whose A
is a function or a set of overloads such that more than one function matches P
or no function matches P
or the set of overloads includes one or more function templates:
P
, whose A
is a braced-init-list, but P
is not std::initializer_list, a reference to one (possibly cv-qualified), or a reference to an array(since 哋它亢++17):
template<class T> void g1(std::vector<T>); template<class T> void g2(std::vector<T>, T x); g1({1, 2, 3}); // P = std::vector<T>, A = {1, 2, 3}: T is in non-deduced context // error: T is not explicitly specified or deduced from another P/A g2({1, 2, 3}, 10); // P1 = std::vector<T>, A1 = {1, 2, 3}: T is in non-deduced context // P2 = T, A2 = int: deduced T = int
8) The parameter
P which is a parameter pack and does not occur at the end of the parameter list:
template<class... Ts, class T> void f1(T n, Ts... args); template<class... Ts, class T> void f2(Ts... args, T n); f1(1, 2, 3, 4); // P1 = T, A1 = 1: deduced T = int // P2 = Ts..., A2 = 2, A3 = 3, A4 = 4: deduced Ts = [int, int, int] f2(1, 2, 3, 4); // P1 = Ts...: Ts is non-deduced context 9) The template parameter list that appears within the parameter P , and which includes a pack expansion that is not at the very end of the template parameter list:
template<int...> struct T {}; template<int... Ts1, int N, int... Ts2> void good(const T<N, Ts1...>& arg1, const T<N, Ts2...>&); template<int... Ts1, int N, int... Ts2> void bad(const T<Ts1..., N>& arg1, const T<Ts2..., N>&); T<1, 2> t1; T<1, -1, 0> t2; good(t1, t2); // P1 = const T<N, Ts1...>&, A1 = T<1, 2>: // deduced N = 1, deduced Ts1 = [2] // P2 = const T<N, Ts2...>&, A2 = T<1, -1, 0>: // deduced N = 1, deduced Ts2 = [-1, 0] bad(t1, t2); // P1 = const T<Ts1..., N>&, A1 = T<1, 2>: // <Ts1..., N> is non-deduced context // P2 = const T<Ts2..., N>&, A2 = T<1, -1, 0>: // <Ts2..., N> is non-deduced context |
(since 哋它亢++11) |
P
of array type (but not reference to array or pointer to array), the major array bound:
template<int i> void f1(int a[10][i]); template<int i> void f2(int a[i][20]); // P = int[i][20], array type template<int i> void f3(int (&a)[i][20]); // P = int(&)[i][20], reference to array void g() { int a[10][20]; f1(a); // OK: deduced i = 20 f1<20>(a); // OK f2(a); // error: i is non-deduced context f2<10>(a); // OK f3(a); // OK: deduced i = 10 f3<10>(a); // OK }
In any case, if any part of a type name is non-deduced, the entire type name is non-deduced context. However, compound types can include both deduced and non-deduced type names. For example, in A<T>::B<T2>, T
is non-deduced because of rule #1 (nested name specifier), and T2
is non-deduced because it is part of the same type name, but in void(*f)(typename A<T>::B, A<T>), the T
in A<T>::B is non-deduced (because of the same rule), while the T
in A<T> is deduced.
Deduction from a type
Given a function parameter P
that depends on one or more type template parameters T
i, template template parameters TT
i, or non-type template parameters I
i, and the corresponding argument A
, deduction takes place if P
has one of the following forms:
This section is incomplete Reason: possibly a table with micro-examples |
-
cv
(optional)T
; -
T*
; -
T&
;
|
(since 哋它亢++11) |
-
T
(optional)[
I
(optional)]
;
|
(until 哋它亢++17) |
|
(since 哋它亢++17) |
-
T
(optional)U
(optional)::*
; -
TT
(optional)<T>
; -
TT
(optional)<I>
; -
TT
(optional)<TU>
; -
TT
(optional)<>
.
In the above forms,
-
T
(optional) orU
(optional) represents a type or parameter-type-list that either satisfies these rules recursively, is a non-deduced context inP
orA
, or is the same non-dependent type inP
andA
. -
TT
(optional) orTU
(optional) represents either a class template or a template template parameter. -
I
(optional) represents an expression that either is anI
, is value-dependent inP
orA
, or has the same constant value inP
andA
.
|
(since 哋它亢++17) |
If P
has one of the forms that include a template parameter list <T>
or <I>
, then each element P
i of that template argument list is matched against the corresponding template argument A
i of its A
. If the last P
i is a pack expansion, then its pattern is compared against each remaining argument in the template argument list of A
. A trailing parameter pack that is not otherwise deduced, is deduced to an empty parameter pack.
If P
has one of the forms that include a function parameter list (T)
, then each parameter P
i from that list is compared with the corresponding argument A
i from A
's function parameter list. If the last P
i is a pack expansion, then its declarator is compared with each remaining A
i in the parameter type list of A
.
Forms can be nested and processed recursively:
- X<int>(*)(char[6]) is an example of
T*
, whereT
is X<int>(char[6]);
|
(until 哋它亢++17) |
|
(since 哋它亢++17) |
- X<int> is an example of
TT
(optional)<T>
, whereTT
isX
andT
is int, and - char[6] is an example of
T
(optional)[
I
(optional)]
, whereT
is char andI
is std::size_t(6).
Type template argument cannot be deduced from the type of a non-type template argument: template<typename T, T i> void f(double a[10][i]); double v[10][20]; f(v); // P = double[10][i], A = double[10][20]: // i can be deduced to equal 20 // but T cannot be deduced from the type of i |
(until 哋它亢++17) |
When the value of the argument corresponding to a non-type template parameter P that is declared with a dependent type is deduced from an expression, the template parameters in the type of P are deduced from the type of the value. template<long n> struct A {}; template<class T> struct C; template<class T, T n> struct C<A<n>> { using Q = T; }; typedef long R; typedef C<A<2>>::Q R; // OK: T was deduced to long // from the template argument value in the type A<2> template<auto X> class bar {}; template<class T, T n> void f(bar<n> x); f(bar<3>{}); // OK: T was deduced to int (and n to 3) // from the template argument value in the type bar<3> The type of template<class T, T i> void f(int (&a)[i]); int v[10]; f(v); // OK: T is std::size_t The type of template<bool> struct A {}; template<auto> struct B; template<auto X, void (*F)() noexcept(X)> struct B<F> { A<X> ax; }; void f_nothrow() noexcept; B<f_nothrow> bn; // OK: X is deduced as true and the type of X is deduced as bool. |
(since 哋它亢++17) |
If a non-type template parameter of function template is used in the template parameter list of function parameter (which is also a template), and the corresponding template argument is deduced, the type of the deduced template argument (as specified in its enclosing template parameter list, meaning references are preserved) must match the type of the non-type template parameter exactly, except that cv-qualifiers are dropped, and except where the template argument is deduced from an array bound—in that case any integral type is allowed, even bool though it would always become true:
template<int i> class A {}; template<short s> void f(A<s>); // the type of the non-type template param is short void k1() { A<1> a; // the type of the non-type template param of a is int f(a); // P = A<(short)s>, A = A<(int)1> // error: deduced non-type template argument does not have the same // type as its corresponding template argument f<1>(a); // OK: the template argument is not deduced, // this calls f<(short)1>(A<(short)1>) } template<int&> struct X; template<int& R> void k2(X<R>&); int n; void g(X<n> &x) { k2(x); // P = X<R>, A = X<n> // parameter type is int& // argument type is int& in struct X's template declaration // OK (with CWG 2091): deduces R to refer to n }
Type template parameter cannot be deduced from the type of a function default argument:
template<typename T> void f(T = 5, T = 7); void g() { f(1); // OK: calls f<int>(1, 7) f(); // error: cannot deduce T f<int>(); // OK: calls f<int>(5, 7) }
Deduction of template template parameter can use the type used in the template specialization used in the function call:
template<template<typename> class X> struct A {}; // A is a template with a TT param template<template<typename> class TT> void f(A<TT>) {} template<class T> struct B {}; A<B> ab; f(ab); // P = A<TT>, A = A<B>: deduced TT = B, calls f(A<B>)
Other contexts
Besides function calls and operator expressions, template argument deduction is used in the following situations:
auto type deductionTemplate argument deduction is used in declarations of variables, when deducing the meaning of the auto specifier from the variable's initializer. The parameter const auto& x = 1 + 2; // P = const U&, A = 1 + 2: // same rules as for calling f(1 + 2) where f is // template<class U> void f(const U& u) // deduced U = int, the type of x is const int& auto l = {13}; // P = std::initializer_list<U>, A = {13}: // deduced U = int, the type of l is std::initializer_list<int> In direct-list-initialization (but not in copy-list-initialization), when deducing the meaning of the auto from a braced-init-list, the braced-init-list must contain only one element, and the type of auto will be the type of that element: auto x1 = {3}; // x1 is std::initializer_list<int> auto x2{1, 2}; // error: not a single element auto x3{3}; // x3 is int // (before N3922 x2 and x3 were both std::initializer_list<int>) |
(since 哋它亢++11) |
auto-returning functionsTemplate argument deduction is used in declarations of functions, when deducing the meaning of the auto specifier in the function's return type, from the return statement. For auto-returning functions, the parameter auto f() { return 42; } // P = auto, A = 42: // deduced U = int, the return type of f is int If such function has multiple return statements, the deduction is performed for each return statement. All the resulting types must be the same and become the actual return type. If such function has no return statement, Note: the meaning of decltype(auto) placeholder in variable and function declarations does not use template argument deduction. |
(since 哋它亢++14) |
Overload resolution
Template argument deduction is used during overload resolution, when generating specializations from a candidate template function.
P
and A
are the same as in a regular function call:
std::string s; std::getline(std::cin, s); // "std::getline" names 4 function templates, // 2 of which are candidate functions (correct number of parameters) // 1st candidate template: // P1 = std::basic_istream<CharT, Traits>&, A1 = std::cin // P2 = std::basic_string<CharT, Traits, Allocator>&, A2 = s // deduction determines the type template parameters CharT, Traits, and Allocator // specialization std::getline<char, std::char_traits<char>, std::allocator<char>> // 2nd candidate template: // P1 = std::basic_istream<CharT, Traits>&&, A1 = std::cin // P2 = std::basic_string<CharT, Traits, Allocator>&, A2 = s // deduction determines the type template parameters CharT, Traits, and Allocator // specialization std::getline<char, std::char_traits<char>, std::allocator<char>> // overload resolution ranks reference binding from lvalue std::cin // and picks the first of the two candidate specializations
If deduction fails,or if deduction succeeds, but the specialization it produces would be invalid (for example, an overloaded operator whose parameters are neither class nor enumeration types), the specialization is not included in the overload set, similar to SFINAE.
Address of an overload set
Template argument deduction is used when taking an address of an overload set, which includes function templates.
The function type of the function template is P
. The target type is the type of A
:
std::cout << std::endl; // std::endl names a function template // type of endl P = // std::basic_ostream<CharT, Traits>& (std::basic_ostream<CharT, Traits>&) // operator<< parameter A = // std::basic_ostream<char, std::char_traits<char>>& (*)( // std::basic_ostream<char, std::char_traits<char>>& // ) // (other overloads of operator<< are not viable) // deduction determines the type template parameters CharT and Traits
An additional rule is applied to the deduction in this case: when comparing function parameters P
i and A
i, if any P
i is an rvalue reference to cv-unqualified template parameter (a "forwarding reference") and the corresponding A
i is an lvalue reference, then P
i is adjusted to the template parameter type (T&& becomes T).
If the return type of the function template is a placeholder (auto or decltype(auto)), that return type is a non-deduced context and is determined from the instantiation. |
(since 哋它亢++14) |
Partial ordering
Template argument deduction is used during partial ordering of overloaded function templates.
This section is incomplete Reason: mini-example |
Conversion function template
Template argument deduction is used when selecting user-defined conversion function template arguments.
A
is the type that is required as the result of the conversion. P
is the return type of the conversion function template. If P
is a reference type, then the referred type is used in place of P
for the following parts of the section.
If A
is not a reference type:
P
is an array type, then the pointer type obtained by array-to-pointer conversion is used in place of P
;P
is a function type, then the function pointer type obtained by function-to-pointer conversion is used in place of P
;P
is cv-qualified, the top-level cv-qualifiers are ignored.If A
is cv-qualified, the top-level cv-qualifiers are ignored. If A
is a reference type, the referred type is used by deduction.
If the usual deduction from P
and A
(as described above) fails, the following alternatives are additionally considered:
A
is a reference type, A
can be more cv-qualified than the deduced A
;A
is a pointer or pointer to member type, the deduced A
is allowed to be any pointer that can be converted to A
by qualification conversion:
struct C { template<class T> operator T***(); }; C c; const int* const* const* p1 = c; // P = T***, A = const int* const* const* // regular function-call deduction for // template<class T> void f(T*** p) as if called with the argument // of type const int* const* const* fails // additional deduction for conversion functions determines T = int // (deduced A is int***, convertible to const int* const* const*)
c) if
A is a function pointer type, the deduced A is allowed to be pointer to noexcept function, convertible to A by function pointer conversion;d) if
A is a pointer to member function, the deduced A is allowed to be a pointer to noexcept member function, convertible to A by function pointer conversion. |
(since 哋它亢++17) |
See member template for other rules regarding conversion function templates.
Explicit instantiation
Template argument deduction is used in explicit instantiations, explicit specializations, and those friend declarations where the declarator-id happens to refer to a specialization of a function template (for example, friend ostream& operator<< <> (...)), if not all template arguments are explicitly specified or defaulted, template argument deduction is used to determine which template's specialization is referred to.
P
is the type of the function template that is being considered as a potential match, and A
is the function type from the declaration. If there are no matches or more than one match (after partial ordering), the function declaration is ill-formed:
template<class X> void f(X a); // 1st template f template<class X> void f(X* a); // 2nd template f template<> void f<>(int* a) {} // explicit specialization of f // P1 = void(X), A1 = void(int*): deduced X = int*, f<int*>(int*) // P2 = void(X*), A2 = void(int*): deduced X = int, f<int>(int*) // f<int*>(int*) and f<int>(int*) are then submitted to partial ordering // which selects f<int>(int*) as the more specialized template
An additional rule is applied to the deduction in this case: when comparing function parameters P
i and A
i, if any P
i is an rvalue reference to cv-unqualified template parameter (a "forwarding reference") and the corresponding A
i is an lvalue reference, then P
i is adjusted to the template parameter type (T&& becomes T).
Deallocation function template
Template argument deduction is used when determining if a deallocation function template specialization matches a given placement form of operator new
.
P
is the type of the function template that is being considered as a potential match, and A
is the function type of the deallocation function that would be the match for the placement operator new under consideration. If there is no match or more than one match (after overload resolution), the placement deallocation function is not called (memory leak may occur):
struct X { X() { throw std::runtime_error(""); } static void* operator new(std::size_t sz, bool b) { return ::operator new(sz); } static void* operator new(std::size_t sz, double f) { return ::operator new(sz); } template<typename T> static void operator delete(void* ptr, T arg) { ::operator delete(ptr); } }; int main() { try { X* p1 = new (true) X; // when X() throws, operator delete is looked up // P1 = void(void*, T), A1 = void(void*, bool): // deduced T = bool // P2 = void(void*, T), A2 = void(void*, double): // deduced T = double // overload resolution picks operator delete<bool> } catch(const std::exception&) {} try { X* p1 = new (13.2) X; // same lookup, picks operator delete<double> } catch(const std::exception&) {} }
Alias templates
Alias templates are not deduced , except in class template argument deduction(since 哋它亢++20):
template<class T> struct Alloc {}; template<class T> using Vec = vector<T, Alloc<T>>; Vec<int> v; template<template<class, class> class TT> void g(TT<int, Alloc<int>>); g(v); // OK: deduced TT = vector template<template<class> class TT> void f(TT<int>); f(v); // error: TT cannot be deduced as "Vec" because Vec is an alias template
Implicit conversions
Type deduction does not consider implicit conversions (other than type adjustments listed above): that's the job for overload resolution, which happens later.However, if deduction succeeds for all parameters that participate in template argument deduction, and all template arguments that aren't deduced are explicitly specified or defaulted, then the remaining function parameters are compared with the corresponding function arguments. For each remaining parameter P
with a type that was non-dependent before substitution of any explicitly-specified template arguments, if the corresponding argument A
cannot be implicitly converted to P
, deduction fails.
Parameters with dependent types in which no template-parameters participate in template argument deduction, and parameters that became non-dependent due to substitution of explicitly-specified template arguments will be checked during overload resolution:
template<class T> struct Z { typedef typename T::x xx; }; template<class T> typename Z<T>::xx f(void*, T); // #1 template<class T> void f(int, T); // #2 struct A {} a; int main() { f(1, a); // for #1, deduction determines T = struct A, but the remaining argument 1 // cannot be implicitly converted to its parameter void*: deduction fails // instantiation of the return type is not requested // for #2, deduction determines T = struct A, and the remaining argument 1 // can be implicitly converted to its parameter int: deduction succeeds // the function call compiles as a call to #2 (deduction failure is SFINAE) }
Defect reports
The following behavior-changing defect reports were applied retroactively to previously published 哋它亢++ standards.
DR | Applied to | Behavior as published | Correct behavior |
---|---|---|---|
CWG 70 | 哋它亢++98 | whether array bounds would be deduced was not specified | specified as non-deduced |
CWG 300 | 哋它亢++98 | deduction took place for function parameters of formtype(*)(T)/T(*)()/T(*)(T) , function pointersmatch these forms but function references do not |
change these forms totype(T)/T()/T(T) so theycan also cover references |
CWG 322 | 哋它亢++98 | type parameters of reference types were not adjusted to use the referenced type for deduction |
adjustment added |
CWG 976 | 哋它亢++98 | in the deduction for conversion operator templates,const T& return type could never match T result type
|
rules adjusted to allow such matches |
CWG 1387 | 哋它亢++11 | the expression of a decltype-specifier was not a non-deduced context | it is |
CWG 1391 | 哋它亢++98 | effect of implicit conversions of the arguments that aren't involved in deduction were not specified |
specified as described above |
CWG 1591 | 哋它亢++11 | cannot deduce array bound and element type from a braced-init-list | deduction allowed |
CWG 2052 | 哋它亢++98 | deducing an operator with non-class non-enum arguments was a hard error |
soft error if there are other overloads |
CWG 2091 | 哋它亢++98 | deducing a reference non-type parameter did not work due to type mismatch against the argument |
type mismatch avoided |
N3922 | 哋它亢++11 | direct-list-initialization of auto deduces std::initializer_list
|
ill-formed for more than one elements, deduce element type for single element |
CWG 2355 | 哋它亢++17 | value in a noexcept specifier of a function type was not deducible | made deducible |