C++ is Lazy: CRTP
In my previous post, Recursion, List Manipulation, and Lazy Evaluation, I wrote about the characteristics of functional programming: The story about lazy evaluation in C++ is short. Sorry to say, but I have forgotten templates. The two advanced techniques, CRTP and expression templates, are based on lazy evaluation.
CRTP
But what does CRTP mean? The acronym CRTP stands for the C++ idiom Curiously Recurring Template Pattern and means a technique in C++ in which a class Derived derives from a class template Base. The key is that Base has Derived as a template argument.
template<class T> class Base{ ... }; class Derived : public Base<Derived>{ ... };
If that is not mind-blowing and how does lazy evaluation kick in? At first lazy evaluation.
As lazy as possible
The key observation for understanding the CRTP idiom is that the instantiation of a method of a class template happens only when needed. Proof?
Modernes C++ Mentoring
Do you want to stay informed: Subscribe.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 |
// lazy.cpp #include <iostream> template<class T> struct Lazy{ void func() { std::cout << "func" << std::endl;} void func2(); // not defined }; int main(){ std::cout << std::endl; Lazy<int> lazy; lazy.func(); std::cout << std::endl; } |
Although the method func2 (line 8) of the class, Lazy is only declared but not defined; the compiler accepts the program. Because I don’t call func2, I need no definition.
That is precisely the property that the CRTP uses because the definition of a method of class templates is only needed if called. The method’s declaration is sufficient for the instantiation of the base class. Therefore, you can implement static polymorphism.
Static Polymorphism
Static polymorphism is quite similar to dynamic polymorphism. But contrary to dynamic polymorphism with virtual methods, the dispatch of the method calls will occur at compile time. Now, we are at the center of the CRTP idiom.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 |
// crtp.cpp #include <iostream> template <typename Derived> struct Base{ void interface(){ static_cast<Derived*>(this)->implementation(); } void implementation(){ std::cout << "Implementation Base" << std::endl; } }; struct Derived1: Base<Derived1>{ void implementation(){ std::cout << "Implementation Derived1" << std::endl; } }; struct Derived2: Base<Derived2>{ void implementation(){ std::cout << "Implementation Derived2" << std::endl; } }; struct Derived3: Base<Derived3>{}; template <typename T> void execute(T& base){ base.interface(); } int main(){ std::cout << std::endl; Derived1 d1; execute(d1); Derived2 d2; execute(d2); Derived3 d3; execute(d3); std::cout << std::endl; } |
I use static polymorphism in the function template execution (lines 29 – 32). I invoke on each argument base the method base.interface. The method Base::interface in lines 7 – 9 is the critical point of the CRTP idiom. The methods dispatch to the implementation of the derived class: static_cast<Derived*>(this)->implementation(). That is possible because the method will be instantiated when called. At this point in time, the derived classes Derived1, Derived2, and Derived3 are fully defined. Therefore, the method Base::interface can use the details of its derived classes. Fascinating is the method Base::implementation (lines 10 – 12). This method plays the role of a default implementation for the static polymorphism for the class Derived3 (line 27).
Here is the output of the program.
Admittedly, the only purpose of the example was to present the mechanic behind the static polymorphism. A convincing example is still missing. Here we are.
Mixins with CRTP
Mixins are a popular concept in mixing classes in new code. Therefore, it’s an often-used technique in Python to change the behavior of a class by using multiple inheritances. Contrary to C++, it is legal in Python to have more than one method definition in a class hierarchy. Python uses that method that is first in the Method Resolution Order (MRO).
You can implement mixins in C++ by using CRTP. A prominent example is the class std::enable_shared_from_this. Using this class, you can create objects that return a std::shared_ptr to themselves. You have to derive your class MySharedClass public from std::enable_shared_from_this. Now, your class MySharedClass has a method shared_from_this for creating std::shared_ptr to its objects. You can read the details about std::enable_shared_from_this in my post Specialities of std::shared_ptr.
An additional typical use-case for mixins is a class that you want to extend with the capability that their instances support the comparison of equality and inequality.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 |
// crtpEquality.cpp #include <iostream> #include <string> template<class Derived> class Equality{}; template <class Derived> bool operator == (Equality<Derived> const& op1, Equality<Derived> const & op2){ Derived const& d1 = static_cast<Derived const&>(op1); Derived const& d2 = static_cast<Derived const&>(op2); return !(d1 < d2) && !(d2 < d1); } template <class Derived> bool operator != (Equality<Derived> const& op1, Equality<Derived> const & op2){ Derived const& d1 = static_cast<Derived const&>(op1); Derived const& d2 = static_cast<Derived const&>(op2); return !(op1 == op2); } struct Apple:public Equality<Apple>{ Apple(int s): size{s}{}; int size; }; bool operator < (Apple const& a1, Apple const& a2){ return a1.size < a2.size; } struct Man:public Equality<Man>{ Man(std::string n): name{n}{} std::string name; }; bool operator < (Man const& m1, Man const& m2){ return m1.name < m2.name; } int main(){ std::cout << std::boolalpha << std::endl; Apple apple1{5}; Apple apple2{10}; std::cout << "apple1 == apple2: " << (apple1 == apple2) << std::endl; Man man1{"grimm"}; Man man2{"jaud"}; std::cout << "man1 != man2: " << (man1 != man2) << std::endl; std::cout << std::endl; } |
I have implemented for the classes Apple and Man the smaller operator (lines 28 and 37). For further reasoning, I will only use the class Man for simplicity. The class Man is publicly derived (lines 32 – 35) from Equality<Man>. I have implemented for classes of the kind Equality<Derived> the equality (lines 9 – 14) and the inequality operator (lines 16 – 21). The inequality operator uses the equality operator (line 20). The equality operator uses the fact that the smaller operator is implemented for Derived (line 13). The equality operator and inequality operator convert its operands: Derived const&: Derived const& d1 = static_cast<Derived const&>(op1).
Now, I can compare Apple and Man for equality and inequality.
What’s next?
In addition to CRTP, expression templates are also based on lazy evaluation. Expression templates are “structures representing a computation at compile time, which are evaluated only as needed to produce efficient code for the entire computation” (https://en.wikipedia.org/wiki/Expression_templates). As needed, that is the point of lazy evaluation; therefore, expression templates are the topic of my next post.
Thanks a lot to my Patreon Supporters: Matt Braun, Roman Postanciuc, Tobias Zindl, G Prvulovic, Reinhold Dröge, Abernitzke, Frank Grimm, Sakib, Broeserl, António Pina, Sergey Agafyin, Андрей Бурмистров, Jake, GS, Lawton Shoemake, Jozo Leko, John Breland, Venkat Nandam, Jose Francisco, Douglas Tinkham, Kuchlong Kuchlong, Robert Blanch, Truels Wissneth, Mario Luoni, Friedrich Huber, lennonli, Pramod Tikare Muralidhara, Peter Ware, Daniel Hufschläger, Alessandro Pezzato, Bob Perry, Satish Vangipuram, Andi Ireland, Richard Ohnemus, Michael Dunsky, Leo Goodstadt, John Wiederhirn, Yacob Cohen-Arazi, Florian Tischler, Robin Furness, Michael Young, Holger Detering, Bernd Mühlhaus, Stephen Kelley, Kyle Dean, Tusar Palauri, Juan Dent, George Liao, Daniel Ceperley, Jon T Hess, Stephen Totten, Wolfgang Fütterer, Matthias Grün, Phillip Diekmann, Ben Atakora, Ann Shatoff, Rob North, Bhavith C Achar, Marco Parri Empoli, Philipp Lenk, Charles-Jianye Chen, Keith Jeffery,and Matt Godbolt.
Thanks, in particular, to Jon Hess, Lakshman, Christian Wittenhorst, Sherhy Pyton, Dendi Suhubdy, Sudhakar Belagurusamy, Richard Sargeant, Rusty Fleming, John Nebel, Mipko, Alicja Kaminska, Slavko Radman, and David Poole.
My special thanks to Embarcadero | |
My special thanks to PVS-Studio | |
My special thanks to Tipi.build | |
My special thanks to Take Up Code | |
My special thanks to SHAVEDYAKS |
Modernes C++ GmbH
Modernes C++ Mentoring (English)
Rainer Grimm
Yalovastraße 20
72108 Rottenburg
Mail: schulung@ModernesCpp.de
Mentoring: www.ModernesCpp.org
Modernes C++ Mentoring,
Leave a Reply
Want to join the discussion?Feel free to contribute!