Run-time type information explained

In computer programming, run-time type information or run-time type identification (RTTI)[1] is a feature of some programming languages (such as C++,[2] Object Pascal, and Ada[3]) that exposes information about an object's data type at runtime. Run-time type information may be available for all types or only to types that explicitly have it (as is the case with Ada). Run-time type information is a specialization of a more general concept called type introspection.

In the original C++ design, Bjarne Stroustrup did not include run-time type information, because he thought this mechanism was often misused.[4]

Overview

In C++, RTTI can be used to do safe typecasts using the dynamic_cast<> operator, and to manipulate type information at runtime using the typeid operator and std::type_info class. In Object Pascal, RTTI can be used to perform safe type casts with the as operator, test the class to which an object belongs with the is operator, and manipulate type information at run time with classes contained in the RTTI unit[5] (i.e. classes: TRttiContext, TRttiInstanceType, etc.). In Ada, objects of tagged types also store a type tag, which permits the identification of the type of these object at runtime. The in operator can be used to test, at runtime, if an object is of a specific type and may be safely converted to it.[6]

RTTI is available only for classes that are polymorphic, which means they have at least one virtual method. In practice, this is not a limitation because base classes must have a virtual destructor to allow objects of derived classes to perform proper cleanup if they are deleted from a base pointer.

Some compilers have flags to disable RTTI. Using these flags may reduce the overall size of the application, making them especially useful when targeting systems with a limited amount of memory.[7]

C++ – typeid

The typeid reserved word (keyword) is used to determine the class of an object at runtime. It returns a reference to std::type_info object, which exists until the end of the program.[8] The use of typeid, in a non-polymorphic context, is often preferred over dynamic_cast<''class_type''> in situations where just the class information is needed, because typeid is always a constant-time procedure, whereas [[dynamic_cast]] may need to traverse the class derivation lattice of its argument at runtime. Some aspects of the returned object are implementation-defined, such as std::type_info::name, and cannot be relied on across compilers to be consistent.

Objects of class std::bad_typeid are thrown when the expression for typeid is the result of applying the unary * operator on a null pointer. Whether an exception is thrown for other null reference arguments is implementation-dependent. In other words, for the exception to be guaranteed, the expression must take the form typeid(*p) where p is any expression resulting in a null pointer.

Example

  1. include
  2. include

class Person ;

class Employee : public Person ;

int main Output (exact output varies by system and compiler):

Person Employee Person* Employee Employee

C++ – dynamic_cast and Java cast

The dynamic_cast operator in C++ is used for downcasting a reference or pointer to a more specific type in the class hierarchy. Unlike the [[static_cast]], the target of the dynamic_cast must be a pointer or reference to class. Unlike [[static_cast]] and C-style typecast (where type check occurs while compiling), a type safety check is performed at runtime. If the types are not compatible, an exception will be thrown (when dealing with references) or a null pointer will be returned (when dealing with pointers).

A Java typecast behaves similarly; if the object being cast is not actually an instance of the target type, and cannot be converted to one by a language-defined method, an instance of java.lang.ClassCastException will be thrown.[9]

Example

Suppose some function takes an object of type A as its argument, and wishes to perform some additional operation if the object passed is an instance of B, a subclass of A. This can be done using dynamic_cast as follows.

  1. include
  2. include
  3. include
  4. include

using namespace std;

class A ;

class B: public A ;

void MyFunction(A& my_a)

int main

Console output: Method specific for B was invoked Method specific for B was invoked Exception std::bad_cast thrown. Object is not of type B

A similar version of MyFunction can be written with pointers instead of references:

void MyFunction(A* my_a)

Object Pascal, Delphi

In Object Pascal and Delphi, the operator is is used to check the type of a class at runtime. It tests the belonging of an object to a given class, including classes of individual ancestors present in the inheritance hierarchy tree (e.g. Button1 is a TButton class that has ancestors: TWinControlTControlTComponentTPersistentTObject, where the latter is the ancestor of all classes). The operator as is used when an object needs to be treated at run time as if it belonged to an ancestor class.

The RTTI unit is used to manipulate object type information at run time. This unit contains a set of classes that allow you to: get information about an object's class and its ancestors, properties, methods and events, change property values and call methods. The following example shows the use of the RTTI module to obtain information about the class to which an object belongs, creating it, and to call its method. The example assumes that the TSubject class has been declared in a unit named SubjectUnit.uses RTTI, SubjectUnit;

procedure WithoutReflection;var MySubject: TSubject;begin MySubject := TSubject.Create; try Subject.Hello; finally Subject.Free; end;end;

procedure WithReflection;var RttiContext: TRttiContext; RttiType: TRttiInstanceType; Subject: TObject;begin RttiType := RttiContext.FindType('SubjectUnit.TSubject') as TRttiInstanceType; Subject := RttiType.GetMethod('Create').Invoke(RttiType.MetaclassType, []).AsObject; try RttiType.GetMethod('Hello').Invoke(Subject, []); finally Subject.Free; end;end;

See also

External links

Notes and References

  1. Web site: Runtime Type Identification . 2000 . Sun Microsystems . Oracle . 16 April 2015 . C++ Programming Guide.
  2. Web site: Language support library [support.rtti] . eel.is . 2021-07-13.
  3. Web site: Object-oriented programming . learn.adacore.com . 2021-07-13.
  4. Web site: Bjarne Stroustrup . Bjarne Stroustrup . A History of C++: 1979—1991 . Bjarne Stroustrup . March 1993 . 50 . 2009-05-18.
  5. Web site: Working with RTTI - RAD Studio. 2021-06-06. docwiki.embarcadero.com.
  6. Book: English, John . Ada 95: The Craft of Object-Oriented Programming . Chapter 15 . 2002-02-22 . 2021-07-13.
  7. Web site: Avoiding RTTI, and support for -fno-rtti in Arm Compiler 6 . Arm Developer . 2021-07-13.
  8. C++ standard (ISO/IEC14882) section 5.2.8 [expr.typeid], 18.5.1 [lib.type.info] – http://cs.nyu.edu/courses/fall11/CSCI-GA.2110-003/documents/c++2003std.pdf
  9. Web site: ClassCastException (Java Platform SE 8).