从零开始学C++之RTTI、dynamic_cast、typeid、类与类之间的关系uml

一、RTTI

Run-time type information (RTTI) is a mechanism that allows the type of an object to be determined during program execution.

There are three main C++ language elements to run-time type information:

  • The dynamic_cast operator. Used for conversion of polymorphic types.
  • The typeid operator. Used for identifying the exact type of an object.
  • The type_info class. Used to hold the type information returned by the typeid operator.  class type_info { public:     virtual ~type_info();     bool operator==(const type_info& rhs) const;     bool operator!=(const type_info& rhs) const;     int before(const type_info& rhs) const;     const char* name() const;     const char* raw_name() const; private:     void *_m_data;     char _m_d_name[1];     type_info(const type_info& rhs);     type_info& operator=(const type_info& rhs);     static const char _Name_base(const type_info *,__type_info_node* __ptype_info_node); }; The result of typeid is a const type_info&. The value is a reference to a type_info object that represents either the type-id or the type of the expression, depending on which form of typeid is used.  为了支持RTTI,为每一个多态类创建一个type_info 对象(静态数据区),并把其地址保存到vtable中的固定位置(一般为第一个位置)(取决于具体编译器实现,标准并没有规定)。 #include <iostream> using namespace std; class Shape { public:     virtual void Draw() = 0;     virtual ~Shape() {} }; class Circle : public Shape { public:     void Draw()     {         cout << "Circle Draw ..." << endl;     } }; class Square : public Shape { public:     void Draw()     {         cout << "Square Draw ..." << endl;     } }; int main(void) {     Shape *p;     Circle c;     p = &c;     p->Draw();     //使用dynamic_cast 的条件     //1、开启运行时类型信息;2、应用在具有多态关系的继承体系上;     if (dynamic_cast<Circle *>(p))     {         cout << "p is point to a Circle object" << endl;         Circle *cp = dynamic_cast<Circle *>(p);     // 安全向下转型         cp->Draw(); //效率没有 p->Draw(); 高     }     else if (dynamic_cast<Square *>(p))     {         cout << "p is point to a Square object" << endl;     }     else     {         cout << "p is point to a Other object" << endl;     }     cout << typeid(*p).name() << endl;     cout << typeid(Circle).name() << endl;     if (typeid(Circle).name() == typeid(*p).name())     {         cout << "p is point to a Circle object" << endl;         ((Circle *)p)->Draw();     }     else if (typeid(Square).name() == typeid(*p).name())     {         cout << "p is point to a Circle object" << endl;         ((Square *)p)->Draw();     }     else     {         cout << "p is point to a Other object" << endl;     }     return 0; }

如上所述,dynamic_cast 和 typeid 操作符 都可以实现运行时类型识别。其中使用dynamic_cast 时需要开启运行时类型信息,在项目-》属性-》C/C++-》语言-》启用运行时类型信息。在使用typeid时需要注意的是返回的是type_info 对象的引用,且type_info 类的拷贝构造函数和赋值运算符都声明为私有,故不能这样写: type_info tf = typeid(Circle); 二、类与类之间的关系 Unified Modeling Language (UML)又称统一建模语言或标准建模语言,是始于1997年一个OMG标准,它是一个支持模型化和软件系统开发的图形化语言。 1、继承(泛化)Generalization

Manager 继承自Employee. 2、关联 Association,单向关联 DirectedAssociation

Order 作为Customer 的成员,如vector<Order>  orders ; 3、聚合 Aggregation

class B class A {  public: B* b_; }; 当A释放时,不负责B的释放,也许B是被共享的。 4、组合 Composition

当Company 释放时要负责Department 的释放,Department 不是共享的。 5、依赖 Dependency

类A依赖于B: 从语义上来上是A use B,偶然的,临时的 B作为A的成员函数参数 B作为A的成员函数的局部变量 A的成员函数调用B的静态方法 比较5种关系: 继承体现的是类与类之间的纵向关系,其他4种体现的是类与类之间的横向关系。 关联强弱 依赖<关联<聚合<组合 继承(A is B) 关联、聚合、组合(A has B) 依赖(A use B)

参考:

C++ primer 第四版 Effective C++ 3rd C++编程规范

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏黑泽君的专栏

对象数组的内存图解

对象数组的概述和案例(掌握) (1)对象数组的概述     数组既可以存储基本数据类型,也可以存储引用类型。它存储引用类型的时候的数组就叫对象数组。 (...

16120
来自专栏java一日一条

Java的内存机制

Java 把内存划分成两种:一种是栈内存,另一种是堆内存。在函数中定义的一些基本类型的变量和对象的引用变量都是在函数的栈内存中分配,当在一段代码块定义一个变量时...

11000
来自专栏Java 源码分析

Java面向对象基础(二)

1.构造器: 构造函数在一个类没有写任何的构造函数的时候,系统会生成一个默认的空参数构造函数,这个构造函数的修饰符就是类的修饰符,当我们定义了一个构造函数,默认...

34970
来自专栏zlmai的专栏

搞清C++中指针、数组、字符串的关系

在一些时候,字符串、指针和数组的关系我们常常搞混,下面通过一些典型问题来展示它们间的关系。

25120
来自专栏向治洪

传统多线程之前如何共享数据

几种方式 线程执行代码相同,使用同一Runnable对象,Runnable对象中有共享数据 线程执行代码不同,将共享数据封装在另一对象中(操作数据的方法也在该...

190100
来自专栏十月梦想

数据类型强制转换

上次提到过数据类型转换在输出的之前(),括号内加入需要转换的类型,这个是临时的转换,下面介绍一下临时转换和永久性数据类型转换。

24030
来自专栏数据结构与算法

1002. 三角形 (

题目描述 输入三角形三边长a,b,c(保证能构成三角形),输出三角形面积。 输入 一行三个用一个空格隔开的实数a,b,c,表示三角形的三条边长。 输出 输出三角...

40760
来自专栏轮子工厂

常见的java面试的基础问题(一) | 看看你会的有多少?

面向对象思想:面向对象是相对于面向过程而言的,面向过程强调的是功能,面向对象强调的是将功能封装进对象,强调具备功能的对象;

7930
来自专栏mySoul

C++命名空间

24120
来自专栏python3

python-其它有用模块1

re模块是文件处理中必不可少的模块,它主要应用于字符串的查找,定位等等,在使用网络爬虫时,即使没有爬虫框架,re模块配合urllib.request模块也可以完...

9120

扫码关注云+社区

领取腾讯云代金券