I wrote a simple example, which estimates average time of calling virtual function, using base class interface and dynamic_cast and call of non-virtual function. Here is it:
It should be noted that the entire purpose of virtual functions is to not have to cast down the inheritance graph. Virtual functions exist so that you can use a derived class instance as though it were a base class. So that more specialized implementations of functions can be called from code that originally called base class versions.
If virtual functions were slower than a safe cast to the derived-class + function call, then C++ compilers would simply implement virtual function calls that way.
So there's no reason to expect dynamic_cast
+call to be faster.
You are just measuring the cost of dynamic_cast<>
. It is implemented with RTTI, that's optional in any C++ compiler. Project + Properties, C/C++, Language, Enable Run-Time Type Info setting. Change it to No.
You'll now get an unsubtle reminder that dynamic_cast<>
can no longer do the proper job. Arbitrarily change it to static_cast<>
to get drastically different results. Key point here is that if you know that an upcast is always safe then static_cast<>
buys you the performance you are looking for. If you don't know for a fact that the upcast is safe then dynamic_cast<>
keeps you out of trouble. It is the kind of trouble that is maddingly hard to diagnose. The common failure mode is heap corruption, you only get an immediate GPF if you are really lucky.
The virtual function call is similar to a function pointer, or if the compiler knows the type, static dispatch. This is constant time.
dynamic_cast
is quite different -- it uses an implementation defined means to determine a type. It is not constant time, may traverse the class hierarchy (also consider multiple inheritance) and perform several lookups. An implementation may use string comparisons. Therefore, the complexity is higher in two dimensions. Real time systems often avoid/discourage dynamic_cast
for these reasons.
More details are available in this document.
The difference is, that you can call the virtual function on any instance that is derived from Base
. The notVirtualCall()
member does not exist within Base
, and cannot be called without first determining the exact dynamic type of the object.
The consequence of this difference is, that the vtable of the base class includes a slot for virtualCall()
, which contains a function pointer to the correct function to call. So, the virtual call simply chases the vtable pointer included as the first (invisible) member of all objects of type Base
, loads the pointer from the slot corresponding to virtualCall()
, and calls the function behind that pointer.
When you do a dynamic_cast<>
, by contrast, the class Base
does not know at compile time what other classes will eventually derive from it. Consequently, it cannot include information within its vtable that eases the resolution of the dynamic_cast<>
. That is the information lack that makes a dynamic_cast<>
more expensive to implement than a virtual function call. The dynamic_cast<>
has to actually search through the inheritance tree of the actual object to check whether the destination type of the cast is found among its bases. That is work that the virtual call avoids.