C++ Templates: Description
C++ has a mechanism called templates to reduce code duplication when supporting numerous data types.
A C++ function or C++ class with functions which operates on integers, float and double data types can be unified with a single template function or class with functions which is flexible enough to use all three data types.
This mechanism in C++ is called the "Template".
C++ templates fall under the category of "meta-programming" and auto code generation although one never sees the code generated.
C++ Template Functions:
In the following example we have a single template represent the code to square a number with the data types int, float and double.
Overloaded functions specified for each data type
14 | double square ( double x) |
30 | cout << i << ": " << ii << endl; |
33 | cout << x << ": " << xx << endl; |
36 | cout << y << ": " << yy << endl; |
Compile: g++ test1.cpp
Run: ./a.out
2: 4
2.2: 4.84
2.2: 4.84
A single template to support all data types
25 | cout << i << ": " << ii << endl; |
27 | xx = square< float >(x); |
28 | cout << x << ": " << xx << endl; |
31 | yy = square< double >(y); |
32 | cout << y << ": " << yy << endl; |
36 | cout << y << ": " << yy << endl; |
Compile: g++ test2.cpp
Run: ./a.out
2: 4
2.2: 4.84
2.2: 4.84
2.2: 4.84
Note:
- The code used in the overloaded C++ functions example above, is repeated for each data type. The templated function is specified once.
- The templated type keyword specifier can be either "class" or "typename":
- template<class T>
- template<typename T>
Both are valid and behave exactly the same. I prefer "typename".
- The templated function works using either the explicit or implicit template expression square<int>(value) or square(value).
- In the template definition, "T" represents the data type. The compiler will generate the type specific functions required. This results in a more compact code base which is easier to maintain.
- The code and logic of the functions only has to be specified once in the templated function and the parameter "T" is used to represent the argument type.
- The template declaration and definition must reside in the same file, typically an include header file.
- A "C" macro function could also perform this purpose: #define square(x) (x * x)
The advantage of the template is that it performs type checking while the macro does not.
Template Specialization:
The following is an extended example of the square function using template specialization to support type string which requires special handling.
Template specialization to support additional data types
14 | string square<string>(string ss) |
25 | cout << i << ": " << ii << endl; |
27 | cout << square<string>(ww) << endl; |
Compile: g++ test.cpp
Run: ./a.out
2: 4
AaaAaa
Note that a specialized template was created to handle the string class.
Template specialization is used when a different and specific implementation is to be used for a specific data type.
Multiple Templated Types:
The following is an example of a template supporting multiple types:
04 | template < typename T, typename U> |
05 | void squareAndPrint(T x, U y) |
09 | cout << "X: " << x << " " << x * x << endl; |
10 | cout << "Y: " << y << " " << y * y << endl; |
18 | squareAndPrint< int , float >(ii, jj); |
Compile: g++ test.cpp
Run: ./a.out
X: 2 4
Y: 2.1 4.41
A single type can only be specified once.
Non-type parameters:
Non-type template parameters provide the ability to pass a constant expression at compile time.
The constant expression may also be an address of a function, object or static class member.
The following is an example of a template function supporting a non-type parameter "count" used for the array size and loop count:
04 | template < typename T, int count> |
09 | for ( int ii=0; ii<count; ii++) |
12 | cout << val[ii] << endl; |
Compile: g++ test.cpp
Run: ./a.out
2.1
3.1
4.1
Specify a default type parameter and default non-type parameter:
04 | template < typename T= float , int count=3> |
07 | for ( int ii=0; ii<count; ii++) |
18 | cout << xx << ": " << multIt<>(xx) << endl;; |
Compile: g++ test.cpp -std=c++0x
Run: ./a.out
2.1: 378.228
Note that multIt<> with no type specified, deferred to the default type: float.
[Potential Pitfall]:
You must specify the compiler argument -std=c++0x to avoid the following error:
test.cpp:5:13: error: default template arguments may not be used in function templates without -std=c++0x or -std=gnu++0x
C++ Template Classes:
The concept of template functions can be extended to template classes.
- A template class takes the form: template <class T> class MyTemplateClass { ... };
- Class template specialization takes the form: template <> class MyTemplateClass <specific-data-type> { ... };
An example of a C++ template class applied to a 2 by 2 matrix:
File:
Matrix2x2.hpp
01 | #ifndef MATRIX_2X2_HPP__ |
02 | #define MATRIX_2X2_HPP__ |
15 | Matrix2x2(T m11, T m12, T m21, T m22); |
20 | int Multiply(Matrix2x2 x) |
26 | Matrix2x2<T>::Matrix2x2(T _m11, T _m12, T _m21, T _m22) |
35 | Matrix2x2<T>::Matrix2x2(T _m) |
44 | Matrix2x2<T>::Matrix2x2() |
53 | Matrix2x2<T>::Add(Matrix2x2 _x) |
56 | sum.m[0][0] = m[0][0] + _x.m[0][0]; |
57 | sum.m[0][1] = m[0][1] + _x.m[0][1]; |
58 | sum.m[1][0] = m[1][0] + _x.m[1][0]; |
59 | sum.m[1][1] = m[1][1] + _x.m[1][1]; |
64 | Matrix2x2<T>::Multiply(Matrix2x2 _x) |
67 | sum.m[0][0] = m[0][0] * _x.m[0][0] + m[0][1] * _x.m[1][0]; |
68 | sum.m[0][1] = m[0][0] * _x.m[0][1] + m[0][1] * _x.m[1][1]; |
69 | sum.m[1][0] = m[1][0] * _x.m[0][0] + m[1][1] * _x.m[1][0]; |
70 | sum.m[1][1] = m[1][0] * _x.m[0][1] + m[1][1] * _x.m[1][1]; |
77 | cout << "|" << m[0][0] << " " << m[0][1] << "|" << endl; |
78 | cout << "|" << m[1][0] << " " << m[1][1] << "|" << endl; |
File:
TestMatrix2x2.cpp
03 | #include "Matrix2x2.hpp" |
07 | int main( int argc, char * argv[]) |
09 | Matrix2x2< int > X(1,2,3,4); |
10 | Matrix2x2< int > Y(5,6,7,8); |
18 | Matrix2x2< int > A = X.Add(Y); |
22 | Matrix2x2< int > B = X.Add(Y); |
Compile: g++ -o TestMatrix2x2 TestMatrix2x2.cpp
Run test:
./TestMatrix2x2
X:
|1 2|
|3 4|
Y:
|5 6|
|7 8|
A:
|6 8|
|10 12|
B:
|19 22|
|43 50|
Static member variables of a template class:
Shown side by side with a regular C++ class for comparison.
Static member variable of a C++ class:
18 | cout << ipri++ << endl; |
31 | cout << aaa.ipub << endl; |
Compile: g++ staticTest.cpp
Run: ./a.out
Output:
1
1
2
Static member variable of a C++ template class:
18 | cout << ipri++ << endl; |
22 | template < class T> T XYZ<T>::ipub = 1; |
23 | template < class T> T XYZ<T>::ipri = 1.2; |
31 | cout << aaa.ipub << endl; |
Compile: g++ staticTemplateTest.cpp
Run: ./a.out
Output:
1
1
1.2
Note that each version or "specialization" (data type) of the class will have its own unique copy of the static member.
All objects of that specialization will share the same static member.
Static member variables are initialized for each specialization/type.
Template template parameters:
4 | template < template < typename T> typename U> |
Template classes and Inheritance:
There are three forms of inheritance which may occur with C++ template classes:
- A templated class may be derived from a regular non-templated C++ base class
- A regular non-templated C++ class can be derived from a templated base class
- A templated class may be derived from another templated class
1) Example of a templated class derived from a regular non-templated C++ class:
File:
Color.hpp (non-templated base class)
04 | enum eColor { none = 0, red, white, blue, yellow, green, black }; |
10 | void setColor(eColor color); |
11 | eColor getColor() { return mColor; }; |
12 | std::string getStrColor(); |
18 | Color::Color(eColor _color) |
23 | void Color::setColor(eColor _color) |
28 | std::string Color::getStrColor() |
File:
Circle.hpp (templated base class)
09 | class Circle : public Color |
12 | Circle(T centerX, T centerY, T radius, eColor color); |
13 | Circle(T centerX, T centerY, T radius); |
30 | Circle<T>::Circle(T _x, T _y, T _radius, eColor _color) |
39 | Circle<T>::Circle(T _x, T _y, T _radius) |
48 | Circle<T>::Circle(T _radius) |
51 | x = static_cast <T>(0); |
52 | y = static_cast <T>(0); |
60 | x = static_cast <T>(0); |
61 | y = static_cast <T>(0); |
62 | radius = static_cast <T>(1); |
68 | return M_PI * radius * radius; |
72 | T Circle<T>::circumference() |
74 | return static_cast <T>(2) * M_PI * radius; |
File:
testCircle.cpp
06 | int main( int argc, char * argv[]) |
08 | Circle< float > circleA(0.0, 0.0, 10.0, white); |
09 | cout << "Area: " << circleA.area() << endl; |
10 | cout << "Color: " << circleA.getStrColor() << endl; |
Compile and run:
g++ -o testCircle testCircle.cpp
./testCircle
Area: 314.159
Color: white
2) Example of a regular non-templated C++ class derived from a templated base class:
(Note that this example uses Color.hpp and Circle.hpp from example one)
File:
Sphere.hpp (derived class)
06 | class Sphere : public Circle< float > |
09 | Sphere( float centerZ, float centerX, float centerY, float radius, eColor color); |
21 | Sphere::Sphere( float _x, float _y, float _z, float _radius, eColor _color) |
22 | : Circle< float >::Circle (_x, _y, _radius, _color) |
27 | Sphere::Sphere( float _radius) |
28 | : Circle::Circle (_radius) |
31 | this ->x = static_cast < float >(0); |
32 | this ->y = static_cast < float >(0); |
33 | this ->z = static_cast < float >(0); |
34 | this ->radius = _radius; |
40 | this ->x = static_cast < float >(0); |
41 | this ->y = static_cast < float >(0); |
42 | this ->z = static_cast < float >(0); |
43 | this ->radius = static_cast < float >(1); |
46 | float Sphere::surfaceArea() |
48 | return static_cast < float >(4) * M_PI * this ->radius * this ->radius; |
55 | return four * M_PI * this ->radius * this ->radius * this ->radius / three; |
Note the use of "this" to show the class dependency. The use of "this" is required.
File:
testSphere.cpp
06 | int main( int argc, char * argv[]) |
08 | Sphere sphereA(0.0, 0.0, 0.0,10.0, blue); |
09 | cout << "Volume: " << sphereA.volume() << endl; |
10 | cout << "Color: " << sphereA.getStrColor() << endl; |
Compile and run:
g++ -o testSphere testSphere.cpp
./testSphere
Volume: 4188.79
Color: blue
3) Example of a templated class derived from another templated class:
(Note that this example uses Color.hpp and Circle.hpp from example one)
File:
Sphere.hpp (derived class)
07 | class Sphere : public Circle<T> |
10 | Sphere(T centerZ, T centerX, T centerY, T radius, eColor color); |
23 | Sphere<T>::Sphere(T _x, T _y, T _z, T _radius, eColor _color) |
24 | : Circle<T>::Circle (_x, _y, _radius, _color) |
30 | Sphere<T>::Sphere(T _radius) |
31 | : Circle<T>::Circle (_radius) |
34 | this ->x = static_cast <T>(0); |
35 | this ->y = static_cast <T>(0); |
36 | this ->z = static_cast <T>(0); |
37 | this ->radius = _radius; |
44 | this ->x = static_cast <T>(0); |
45 | this ->y = static_cast <T>(0); |
46 | this ->z = static_cast <T>(0); |
47 | this ->radius = static_cast <T>(1); |
51 | T Sphere<T>::surfaceArea() |
53 | return static_cast <T>(4) * M_PI * this ->radius * this ->radius; |
61 | return four * M_PI * this ->radius * this ->radius * this ->radius / three; |
Note the use of "this" to show the class dependency. The use of "this" is required.
File:
testSphere.cpp
06 | int main( int argc, char * argv[]) |
08 | Sphere< float > sphereA(0.0, 0.0, 0.0,10.0, blue); |
09 | cout << "Volume: " << sphereA.volume() << endl; |
10 | cout << "Color: " << sphereA.getStrColor() << endl; |
Compile and run:
g++ -o testSphere testSphere.cpp
./testSphere
Volume: 4188.79
Color: blue
Template Argument Deduction:
Template Argument Deduction refers to the deduction of a template argument type from its implied use.
There are three types of C++ argument deduction:
- Template argument deduction from the function call expression
- Template argument deduction from the address of a member function template specialization
- Template argument deduction from specialization of a member function template
Links:

Books: