C++——C++中的类型识别

1.在面向对象中可能出现下面的情况

(1)基类指针指向子类对象

(2)基类引用成为子类对象的别名

静态类型------变量(对象)自身的类型

动态类型------指针(引用)所指向对象的实际类型

基类指针是否可以强制类型转换为子类指针取决于动态类型,若指针指向子类就可以转换,指向父类就无法转换,所以C++中如何得到动态类型

解决方案:利用多态

  • 在基类中定义虚函数返回具体的类型信息

  • 所有的派生类都必须实现类型相关的虚函数

  • 每个类中的类型虚函数都需要不同的实现

    #include<iostream>
    #include<string>
    using namespace std;
    class Base {
    public:
    virtual string type() {
    return "Base";
    }
    };
    class Derived :public Base {
    public:
    string type() {
    return "Derived";
    }
    void print() {
    cout << "Derived" << endl;
    }
    };
    class Child :public Base {
    public:
    string type() {
    return "Child";
    }
    };
    void test(Base* b) {
    // Derived* d = static_cast<Derived*>(b); 危险的转换方式
    cout << b->type() << endl;
    if (b->type() == "Derived") {
    Derived* d = static_cast<Derived*>(b);
    d->print();
    }
    cout << dynamic_cast<Derived*>(b) << endl; //dynamic_cast只能显示转换是否成功,但是无法知道转换后的结果
    }
    int main() {
    Base b;
    Derived d;
    Child c;
    test(&b);
    test(&d);
    test(&c);
    }

运行结果:

Base

0000000000000000

Derived

Derived

00000014E231F698

Child

0000000000000000

但是上述方式的缺陷

  • 必须从基类开始提供类型虚函数
  • 所有的派生类都必须重写类型虚函数
  • 每个派生类的类型名必须唯一

C++提供了typeid关键字用于获取类型信息

  • typeid关键字返回对应参数的类型信息
  • typeid返回一个type_info类对象
  • 当typeid的参数为NULL时将抛出异常

注意事项

  • 当参数为类型时:返回静态类型信息

  • 当参数为变量时:不存在虚函数表------返回静态类型信息;存在虚函数表------返回动态类型信息

    #include<iostream>
    #include<string>
    #include<typeinfo>
    using namespace std;
    class Base {
    public:

    };
    class Derived :public Base {
    public:
    void printf() {
    cout << "Derived" << endl;
    }
    };

    void test(Base* b) {
    const type_info& tb = typeid(*b);
    cout << tb.name() << endl;
    }
    int main() {
    int i = 0;
    const type_info& tiv = typeid(i);
    const type_info& tii = typeid(int);
    cout << (tiv == tii) << endl; //1
    Base b;
    Derived d;
    test(&b); //class Base
    test(&d); //class Base
    return 0;
    }

由于b指针所指向的对象没有虚函数表,所有返回静态类型信息

复制代码
#include<iostream>
#include<string>
#include<typeinfo>
using namespace std;
class Base {
public:
	virtual ~Base() {

	}
};
class Derived :public Base {
public:
	void printf() {
		cout << "Derived" << endl;
	}
};

void test(Base* b) {
	const type_info& tb = typeid(*b);
	cout << tb.name() << endl;
}
int main() {
	int i = 0;
	const type_info& tiv = typeid(i);
	const type_info& tii = typeid(int);
	cout << (tiv == tii) << endl; //1
	Base b;
	Derived d;
	test(&b); //class Base
	test(&d); //class Derived
	return 0;
}

存在虚函数表------返回动态类型信息

相关推荐
SilentSamsara27 分钟前
Python 环境搭建完整指南:从下载安装到运行第一个程序
开发语言·python
小短腿的代码世界40 分钟前
Qt文件系统与IO深度解析:从QFile到异步文件操作
开发语言·qt
智者知已应修善业1 小时前
【51单片机按键调节占空比3位数码管显示】2023-8-24
c++·经验分享·笔记·算法·51单片机
harder3212 小时前
RMP模式的创新突破
开发语言·学习·ios·swift·策略模式
jinanwuhuaguo2 小时前
OpenClaw工程解剖——RAG、向量织构与“记忆宫殿”的索引拓扑学(第十三篇)
android·开发语言·人工智能·kotlin·拓扑学·openclaw
Rust研习社2 小时前
使用 Axum 构建高性能异步 Web 服务
开发语言·前端·网络·后端·http·rust
徐某人..3 小时前
基于i.MX6ULL平台的智能网关系统开发
arm开发·c++·单片机·qt·物联网·学习·arm
无敌秋4 小时前
# C++ 简单工厂模式实战指南
c++·简单工厂模式
淘矿人4 小时前
从0到1:用Claude启动你的第一个项目
开发语言·人工智能·git·python·github·php·pygame