c++ typeid和type_index

2024-04-02 12:32
文章标签 c++ type index typeid

本文主要是介绍c++ typeid和type_index,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

typeid

查询类型的信息。
用于必须知晓多态对象的动态类型的场合以及静态类型鉴别。

使用方法

头文件: #include<typeinfo>
typeid(类型) 或者typeid(表达式)

  1. 指代一个表示 类型 的 std::type_info 对象。若 类型 为引用类型,则结果所指代的 std::type_info 对象表示被引用的类型。
  2. 检验表达式
    a) 若 表达式 为标识某个多态类型(即声明或继承至少一个虚函数的类)对象的泛左值表达式,则 typeid 表达式对该表达式求值,然后指代表示该表达式动态类型的 std::type_info 对象。若该泛左值表达式为通过对一个指针应用一元 * 运算符所得,且该指针为空指针值,则抛出 std::bad_typeid 类型或从 std::bad_typeid 派生的类型的异常。
    b) 若 表达式 不是多态类型的泛左值表达式,则 typeid 不对该表达式求值,而它所指代的 std::type_info 对象表示该表达式的静态类型。不进行左值到右值、数组到指针或函数到指针转换。然而对于纯右值参数,(形式上)要进行临时量实质化:typeid 确定其结果对象的类型。 (C++17 起)
    在所有情况下,typeid 都忽略 cv 限定符(即 typeid(T) == typeid(const T))

注意

应用于多态类型的表达式时,typeid 表达式的求值可能涉及运行时开销(虚表查找),其他情况下 typeid 表达式都在编译时解决。

typeid 所指代的对象的析构函数是否在程序结束时执行是未指明的。
不保证同一类型上的 typeid 表达式的所有求值都指代同一个 std::type_info 实例,不过这些 type_info 对象的 std::type_info::hash_code 相同,其 std::type_index 也相同。

const std::type_info& ti1 = typeid(A);
const std::type_info& ti2 = typeid(A);assert(&ti1 == &ti2); // 不保证
assert(ti1.hash_code() == ti2.hash_code()); // 保证
assert(std::type_index(ti1) == std::type_index(ti2)); // 保证

示例代码

#include <iostream>
#include <string>
#include <typeinfo>struct Base {}; // 非多态
struct Derived : Base {};struct Base2 { virtual void foo() {} }; // 多态
struct Derived2 : Base2 {};int main() {int myint = 50;std::string mystr = "string";double *mydoubleptr = nullptr;std::cout << "myint has type: " << typeid(myint).name() << '\n'<< "mystr has type: " << typeid(mystr).name() << '\n'<< "mydoubleptr has type: " << typeid(mydoubleptr).name() << '\n';// std::cout << myint 为多态类型的泛左值表达式;求值const std::type_info& r1 = typeid(std::cout << myint);std::cout << '\n' << "std::cout<<myint has type : " << r1.name() << '\n';// std::printf() 不是多态类型的泛左值表达式;不求值const std::type_info& r2 = typeid(std::printf("%d\n", myint));std::cout << "printf(\"%d\\n\",myint) has type : " << r2.name() << '\n';// 非多态左值时为静态类型Derived d1;Base& b1 = d1;std::cout << "reference to non-polymorphic base: " << typeid(b1).name() << '\n';Derived2 d2;Base2& b2 = d2;std::cout << "reference to polymorphic base: " << typeid(b2).name() << '\n';try {// 解引用空指针:对于非多态表达式 OKstd::cout << "mydoubleptr points to " << typeid(*mydoubleptr).name() << '\n'; // 解引用空指针:对多态左值则不行Derived2* bad_ptr = nullptr;std::cout << "bad_ptr points to... ";std::cout << typeid(*bad_ptr).name() << '\n';} catch (const std::bad_typeid& e) {std::cout << " caught " << e.what() << '\n';}
}

输出:

myint has type: int
mystr has type: std::basic_string<char, std::char_traits<char>, std::allocator<char> >
mydoubleptr has type: double*
50
std::cout<<myint has type : std::basic_ostream<char, std::char_traits<char> >
printf("%d\n",myint) has type : int
reference to non-polymorphic base: Base
reference to polymorphic base: Derived2
mydoubleptr points to double
bad_ptr points to...  caught std::bad_typeid

样例2 lambda函数type_index

#include <iostream>
#include <functional>
#include <typeinfo>
#include <typeindex>
using namespace std;
class Base {};
class Derived: public Base {};void make_number(int a, int b) {std::cout << "a:" << a << " b:" << b << std::endl;
}int main() {std::cout << "class Base:" << (*(&typeid(Base))).name() << std::endl;std::cout << "class Derived:" << (*(&typeid(Derived))).name() << std::endl;// pointer 类型 typeinfostd::cout << "&make_number:" << typeid(&make_number).name() << std::endl;// 普通类型 typeinfostd::cout << "make_number:" << typeid(make_number).name() << std::endl;const std::type_info* make_number_info = &typeid(&make_number);typedef void (*func)(int, int);auto & tid = typeid(func); // auto  tid = typeid(func); // 报错,calling a provate constructstd::cout << "func :" << typeid(func).name() << std::endl;bool number = std::type_index(*(make_number_info)) == std::type_index(tid);std::cout << "bool type_index :" << number << std::endl;}

可能的输出:

class Base:4Base
class Derived:7Derived
&make_number:PFviiE
make_number:FviiE
func :PFviiE
bool type_index :1

这篇关于c++ typeid和type_index的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/869987

相关文章

Windows下C++使用SQLitede的操作过程

《Windows下C++使用SQLitede的操作过程》本文介绍了Windows下C++使用SQLite的安装配置、CppSQLite库封装优势、核心功能(如数据库连接、事务管理)、跨平台支持及性能优... 目录Windows下C++使用SQLite1、安装2、代码示例CppSQLite:C++轻松操作SQ

C++中RAII资源获取即初始化

《C++中RAII资源获取即初始化》RAII通过构造/析构自动管理资源生命周期,确保安全释放,本文就来介绍一下C++中的RAII技术及其应用,具有一定的参考价值,感兴趣的可以了解一下... 目录一、核心原理与机制二、标准库中的RAII实现三、自定义RAII类设计原则四、常见应用场景1. 内存管理2. 文件操

C++中零拷贝的多种实现方式

《C++中零拷贝的多种实现方式》本文主要介绍了C++中零拷贝的实现示例,旨在在减少数据在内存中的不必要复制,从而提高程序性能、降低内存使用并减少CPU消耗,零拷贝技术通过多种方式实现,下面就来了解一下... 目录一、C++中零拷贝技术的核心概念二、std::string_view 简介三、std::stri

C++高效内存池实现减少动态分配开销的解决方案

《C++高效内存池实现减少动态分配开销的解决方案》C++动态内存分配存在系统调用开销、碎片化和锁竞争等性能问题,内存池通过预分配、分块管理和缓存复用解决这些问题,下面就来了解一下... 目录一、C++内存分配的性能挑战二、内存池技术的核心原理三、主流内存池实现:TCMalloc与Jemalloc1. TCM

C++ 函数 strftime 和时间格式示例详解

《C++函数strftime和时间格式示例详解》strftime是C/C++标准库中用于格式化日期和时间的函数,定义在ctime头文件中,它将tm结构体中的时间信息转换为指定格式的字符串,是处理... 目录C++ 函数 strftipythonme 详解一、函数原型二、功能描述三、格式字符串说明四、返回值五

C++作用域和标识符查找规则详解

《C++作用域和标识符查找规则详解》在C++中,作用域(Scope)和标识符查找(IdentifierLookup)是理解代码行为的重要概念,本文将详细介绍这些规则,并通过实例来说明它们的工作原理,需... 目录作用域标识符查找规则1. 普通查找(Ordinary Lookup)2. 限定查找(Qualif

C/C++ chrono简单使用场景示例详解

《C/C++chrono简单使用场景示例详解》:本文主要介绍C/C++chrono简单使用场景示例详解,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友... 目录chrono使用场景举例1 输出格式化字符串chrono使用场景China编程举例1 输出格式化字符串示

C++/类与对象/默认成员函数@构造函数的用法

《C++/类与对象/默认成员函数@构造函数的用法》:本文主要介绍C++/类与对象/默认成员函数@构造函数的用法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录名词概念默认成员函数构造函数概念函数特征显示构造函数隐式构造函数总结名词概念默认构造函数:不用传参就可以

C++类和对象之默认成员函数的使用解读

《C++类和对象之默认成员函数的使用解读》:本文主要介绍C++类和对象之默认成员函数的使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、默认成员函数有哪些二、各默认成员函数详解默认构造函数析构函数拷贝构造函数拷贝赋值运算符三、默认成员函数的注意事项总结一

C/C++中OpenCV 矩阵运算的实现

《C/C++中OpenCV矩阵运算的实现》本文主要介绍了C/C++中OpenCV矩阵运算的实现,包括基本算术运算(标量与矩阵)、矩阵乘法、转置、逆矩阵、行列式、迹、范数等操作,感兴趣的可以了解一下... 目录矩阵的创建与初始化创建矩阵访问矩阵元素基本的算术运算 ➕➖✖️➗矩阵与标量运算矩阵与矩阵运算 (逐元