Posted by Rico's Nerd Cluster on February 10, 2023
typename Is Required By “Dependent Names”
Dependent names are names that are dependent on the type of a template parameter. To instantiate instances of them, or to simply use them as types, one needs to use the keyword typename.
Consider the below snippet:
1
2
3
4
5
6
7
8
9
10
11
12
13
// namespace MyClass { // can't be namespace, because typename T expects a type, such as class/structstructMyClass{structconstant{staticconstexprintvalue=42;};};template<typenameT>voidexampleFunction(){typenameT::constantmyConstant;// No 'template' keyword needed// value is a static member, and it's accessible to the class itselfstd::cout<<"Value: "<<myConstant.value<<std::endl;}
:: means type within a type
typename T can’t take in a namespace. It has to be a type. It is to create an object of the type T
typename T::constant myConstant can access static members, like myConstant.value
Non-Type Template Parameters
Non Type template parameters are compile time constants that customizes a struct. Some notes include:
Foo<9> is a different instantiation of Foo<10>
1
2
3
4
5
6
7
8
9
10
11
12
template<intN>structFoo{staticconstinti=N;};intmain(){Foo<9>f;std::cout<<f.i<<std::endl;// Print the static class memberstd::cout<<Foo<10>::i<<std::endl;}
Specialized Template Function To Provide One Implementation
#include<iostream>
#include<type_traits> // For std::remove_pointertemplate<classT>voidf(T&x){// Resolve the base type of T if T is a pointerusingBaseType=typenamestd::remove_pointer<T>::type;// So BaseType is FTemplates// template function variable() needs to be accessed with an explicit "template"// BaseType::template is needed to indicate constant is a nested template inside Basetypex->templatevariable<typenameBaseType::templateconstant<3>>();}structFTemplates{// Nested templatetemplate<intN>structconstant{staticconstexprintvalue=N;};// Member templatetemplate<typenameT>voidvariable(){std::cout<<"variable: "<<T::value<<std::endl;}};intmain(){FTemplatesf_tmp;FTemplates*f_ptr=&f_tmp;// Call f with a pointer to f_templatesf(f_ptr);// Outputs: variable: 3return0;}
Concepts For T [C++ 20]
When enforcing type constraints, in C++ 17, one needs to use std::enable_if. That results in a longer template signature. In C++ 20, one just needs to use Concepts
#include<iostream>
#include<type_traits>
#include<concepts>template<std::integralT>voidprint_20(Tx){std::cout<<x<<" is an integer\n";}template<typenameT>typenamestd::enable_if<std::is_integral<T>::value,void>::typeprint_17(Tx){std::cout<<x<<" is an integer\n";}intmain(){// print_17("123"); // See compiler error: no matching function for call to ‘print(const char [4])’print_17(123);// can compile fineprint_20(456);}
✅ More readable.
✅ More efficient (compiler checks constraints upfront).
In C++17, std::enable_ifis checked during template instantiation, where errors are usually versbose.
In C++20, concepts are checked before template instantiation. Errors are clearer.
The requires Clause [C++20]
Puts constraints on template parameters. It takes a “boolean concept”
compile-time predicatestd::integral<T> is equivalent to concept integral = std::is_integral_v<T>;
1
2
3
4
5
6
7
8
9
10
11
template<typenameT>Tadd(Ta,Tb)requiresstd::integral<T>{returna+b;}// Can be before the function signature, or aftertemplate<typenameT>requiresstd::integral<T>Tadd(Ta,Tb){returna+b;}
One note is that C++ standard has NOT specified if an template alias is the same as the template itself. On some compilers (e.g., x86-64 clang 10.0.1), the below code snippet can compile fine. But on some others, A and B will be treated as the same template, so the template specialization would yield errors (x86-64 gcc 14.1)