“Virtual functions” are functions that ensure that the correct function is called for an object, regardless of the expression used to make the function call.
Suppose a base class contains a function declared as virtual and a derived class defines the same function. The function from the derived class is invoked for objects of the derived class, even if it is called using a pointer or reference to the base class. The following example shows a base class that provides an implementation of the PrintBalance
function:
class Account
{
public:
Account( double d ); // Constructor.
virtual double GetBalance(); // Obtain balance.
virtual void PrintBalance(); // Default implementation.
private:
double _balance;
};
// Implementation of constructor for Account.
double Account::Account( double d )
{
_balance = d;
}
// Implementation of GetBalance for Account.
double Account::GetBalance()
{
return _balance;
}
// Default implementation of PrintBalance.
void Account::PrintBalance()
{
cerr << "Error. Balance not available for base type."
<< endl;
}
Two derived classes, CheckingAccount
and SavingsAccount
, can be created as follows:
class CheckingAccount : public Account
{
public:
void PrintBalance();
};
// Implementation of PrintBalance for CheckingAccount.
void CheckingAccount::PrintBalance()
{
cout << "Checking account balance: " << GetBalance();
}
class SavingsAccount : public Account
{
public:
void PrintBalance();
};
// Implementation of PrintBalance for SavingsAccount.
void SavingsAccount::PrintBalance()
{
cout << "Savings account balance: " << GetBalance();
}
The PrintBalance
function in the derived classes is virtual because it is declared as virtual in the base class, Account
. To call virtual functions such as PrintBalance
, code such as the following can be used:
// Create objects of type CheckingAccount and SavingsAccount.
CheckingAccount *pChecking = new CheckingAccount( 100.00 );
SavingsAccount *pSavings = new SavingsAccount( 1000.00 );
// Call PrintBalance using a pointer to Account.
Account *pAccount = pChecking;
pAccount->PrintBalance();
// Call PrintBalance using a pointer to Account.
pAccount = pSavings;
pAccount->PrintBalance();
In the preceding code, the calls to PrintBalance
are identical, except for the object pAccount
points to. Because PrintBalance
is virtual, the version of the function defined for each object is called. The PrintBalance
function in the derived classes CheckingAccount
and SavingsAccount
“override” the function in the base class Account
.
If a class is declared that does not provide an overriding implementation of the PrintBalance
function, the default implementation from the base class Account
is used.
Functions in derived classes override virtual functions in base classes only if their type is the same. A function in a derived class cannot differ from a virtual function in a base class in its return type only; the argument list must differ as well.
When calling a function using pointers or references, the following rules apply:
The following example shows how virtual and nonvirtual functions behave when called through pointers:
#include <iostream.h>
// Declare a base class.
class Base
{
public:
virtual void NameOf(); // Virtual function.
void InvokingClass(); // Nonvirtual function.
};
// Implement the two functions.
void Base::NameOf()
{
cout << "Base::NameOf\n";
}
void Base::InvokingClass()
{
cout << "Invoked by Base\n";
}
// Declare a derived class.
class Derived : public Base
{
public:
void NameOf(); // Virtual function.
void InvokingClass(); // Nonvirtual function.
};
// Implement the two functions.
void Derived::NameOf()
{
cout << "Derived::NameOf\n";
}
void Derived::InvokingClass()
{
cout << "Invoked by Derived\n";
}
void main()
{
// Declare an object of type Derived.
Derived aDerived;
// Declare two pointers, one of type Derived * and the other
// of type Base *, and initialize them to point to aDerived.
Derived *pDerived = &aDerived;
Base *pBase = &aDerived;
// Call the functions.
pBase->NameOf(); // Call virtual function.
pBase->InvokingClass(); // Call nonvirtual function.
pDerived->NameOf(); // Call virtual function.
pDerived->InvokingClass(); // Call nonvirtual function.
}
The output from this program is:
Derived::NameOf
Invoked by Base
Derived::NameOf
Invoked by Derived
Note that regardless of whether the NameOf
function is invoked through a pointer to Base
or a pointer to Derived
, it calls the function for Derived
. It calls the function for Derived
because NameOf
is a virtual function, and both pBase
and pDerived
point to an object of type Derived
.
Because virtual functions are called only for objects of class types, you cannot declare global or static functions as virtual.
The virtual keyword can be used when declaring overriding functions in a derived class, but it is unnecessary; overrides of virtual functions are always virtual.
Virtual functions in a base class must be defined unless they are declared using the pure-specifier. (For more information about pure virtual functions, see Abstract Classes.)
The virtual function-call mechanism can be suppressed by explicitly qualifying the function name using the scope-resolution operator (::). Consider the preceding example. To call PrintBalance
in the base class, use code such as the following:
CheckingAccount *pChecking = new CheckingAccount( 100.00 );
pChecking->Account::PrintBalance(); // Explicit qualification.
Account *pAccount = pChecking; // Call Account::PrintBalance
pAccount->Account::PrintBalance(); // Explicit qualification.
Both calls to PrintBalance
in the preceding example suppress the virtual function-call mechanism.