Alternatives to virtual functions include the NVI idiom (as an example of the Template Method design pattern) and various forms of the Strategy design pattern.
Suppose we want to implement a member function named healthValue, which will calculate the health value for different characters in differen ways in a video game. Declaring healthValue virtual seems the obvious way to design things:
|  |  | 
Actually, except from this obvious design, there exists some alternatives. To name a few:
- Use the non-virtual interface idiom (NVI idiom), a form of the Template Method design pattern that wraps public non-virtual member functions around less accessible virtual functoins
- Use the Strategy design pattern, specifically:
- Replace virtual function with function pointer data members - a stripped-down manifestation of Strategy design pattern
- Replace virtual function with tr1::function data members, which allows use of any callable entity with a signature compatible with what we need - a more general form of the stripped-down representation of Strategy design pattern
- Replace virtual functions in one hierarchy with virtual functions in another hierarchy - the conventional implementation of the Strategy design pattern
 
Let’s take a look at the pros and cons of these alternatives.
The template method pattern via the non-virtual interface idiom
The non-virtual interface (NVI) idiom argues that clients should call private virtual functions indirectly through public non-virtual member functions, which act like a wrapper around the virtual functions. This is a particular manifestation of the more general design pattern known as Template Method1.
|  |  | 
The NVI idiom involves derived classes redefining private virtual functions that they can’t call - this seemingly contradictory rule is allowed by C++ and actually makes perfect sense: redefining a virtual function specifies how something is to be done, while calling a virtual function specifies when it will be done, and these two concerns are independent.
However, it’s not strictly necessary to declare the virtual functions private: in the cases where the derived class implementations of a virtual function are expected to invoke their base class counterparts, we have to declare the virtuals protected. Sometimes a virtual function even has to be public (e.g., destructors in polymorphic base classes, item 7), but then the NVI idiom can’t really be applied.
The advantage of the NVI idiom is in the ““do ‘before’ stuff” and “do ‘after’ stuff” part in the code above, which enable the wrapper abilities to ensure that before a virtual function is called, the proper context is set up (e.g., locking a mutex, making a log entry, verifying the class invariants and function pereconditions, etc), and after the call is over, the context is cleaned up (e.g., unlocking a mutex, verifying function postconditions, etc). If letting clients call virtual functions directly, there’s no good way to do these stuff.
The strategy pattern
The NVI idiom is still using virtual functions to calculate a character’s health. A more dramatic design assertion, such as the strategy pattern, says that calculating a character’s health is independent of the character’s type - the calculation need not to be part of the character.
It is worth noting that, due to its being outside the GameCharacter hierarchy, the calculation has no special access to the internal parts of the object whose health it’s calculating, so this design pattern works only if a character’s health can be calculated based purely on information available through the character’s public interface. Thus it may require the class’s encapsulation to be weakened (e.g., make the non-member functions to be friend, or offer public accessor functions for class implementation that is previously hidden).
The strategy pattern via function pointers
|  |  | 
The advantage of this approach is some interesting flexibility:
- different instances of the same character type can have different health calculation functions installed
- health calculation functions for a particular character may be changed at runtime. For example, GameCharactermight offer a member functionsetHealthCalculatorthat allowed replacement of the current health calculation function.
The strategy pattern via TR1::function
To generalized the approach above, we could replace the function pointer healthFunc with an object of type tr1::function. As item 54 explains, such objects may hold any callable entity (e.g., function pointer, function object, member function pointer) whose signature is compatible2 with the given target signature:
|  |  | 
The advantage of this approach is that clients may have more flexibility in specifying health calculation functions with various compatible callable entities:
|  |  | 
To use these callable entities:
|  |  | 
Here, the purpose of calling tr1::bind is that:
- it adapts the GameLevel::healthmember function (which takes two parameters: an implicitGameLevelparameter thatthispoints to, as well as another reference to aGameCharacterparameter) into the expected function signature (which should only take a single paramter: theGameCharacter)
- by passing _1, it specifies that when callingGameLevel::healthforebg2, always usecurrentLevelas the first parameter ofGameLevelobject
The “Classic” strategy pattern
The conventional approach to Strategy design pattern would be to make the health-calculation function a virtual member function of a separate health-calculation hierarchy with root being HealthCalcFunc, so that different health calculation algorithm could be implemented in the derived classes in this inheritance hierarchy. Each object of type GameCharacter just contains a pointer to an object from the HealthCalcFunc hierarchy:
|  |  | 
- 
Template Method design pattern has nothing to do with C++ template. ↩︎ 
- 
Here, the “target signature” is “function taking a reference to a const GameCharacterand returning anint”, so any callable entity whose parameter can be implicitly converted to aconst GameCharacter&and whose return type can be implicitly converted to anintis compatible with an object of the declaredtr1::functiontype. ↩︎