C++:获取类成员的偏移量

2024-06-06 22:08
文章标签 c++ 获取 成员 偏移量

本文主要是介绍C++:获取类成员的偏移量,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

出自:http://blog.163.com/henan_lujun/blog/static/19538333201078111820803/

第一种方式是MFC里使用广泛的宏对空对象成员取地址

#define OFFSET(structure, member) ((int)&((structure*)0)->member);

正如我们平时通过某对象的地址指针访问某个成员变量一样,这里只是强制使用0作为该地址,但区别是并没有通过该地址去访问成员变量,而只是用&操作符来获取该成员变量的地址,所以不会出现访问违规的情况。

object member address = object address + member offset

因此当对象地址为0时,对对象成员取地址得到的就是该成员在对象中的偏移量。只是对虚基类中的成员无法通过此种形式获得其偏移量——一般的编译器都需要通过从虚表中获得虚基类部分在对象中的偏移量,而空对象其vptr无效,所以无法获得虚表,因此也无法获得虚基类部分的偏移量。

    Point3D *point = 0;

    cout << "&NULLPointer._x " << &(point->_x<< endl;

    cout << "&NULLPointer._y " << &(point->_y<< endl;

    cout << "&NULLPointer._z " << &(point->_z<< endl;

输出:

&NULLPointer._x 0

&NULLPointer._y 0x4

&NULLPointer._z 0x8

 

 

第二种方式是通过类成员指针获得成员偏移量

另一种方式是通过域操作符取成员变量的地址。例如一个类Testint 型成员变量x,则可以通过int Test::* pOffset = &Test::x 获得该偏移量,然后通过

int nOffset = reinterpret_cast<int>(*(void**)(&pOffset))将其转化为整型量。

int Point3D::* offsetx = &Point3D::_x;

int Point3D::* offsety = &Point3D::_y;

int Point3D::* offsetz = &Point3D::_z;

 

int nOffsetx = reinterpret_cast<int>(*(void**)(&offsetx));

int nOffsety = reinterpret_cast<int>(*(void**)(&offsety));

int nOffsetz = reinterpret_cast<int>(*(void**)(&offsetz));

cout << "nOffsetx= reinterpret_cast<int>(*(void**)(&offsetx))"<< nOffsetx <<endl;

cout << "nOffset = reinterpret_cast<int>(*(void**)(&offsety))" << nOffsety <<endl;

cout << "nOffsetz= reinterpret_cast<int>(*(void**)(&offsetz))"<< nOffsetz <<endl;

 

输出:

int nOffsetx = reinterpret_cast<int>(*(void**)(&offsetx)) =  0

int nOffsety = reinterpret_cast<int>(*(void**)(&offsety)) =  4

int nOffsetz = reinterpret_cast<int>(*(void**)(&offsetz)) =  8

 

上述的输出方式过于麻烦,如果用printf则可以直接作为整数输出。

    printf ( "int Point3D::* offsetx = %d\n",offsetx );

    printf ( "int Point3D::* offsety = %d\n",offsety );

    printf ( "int Point3D::* offsetz = %d\n",offsetz );

 

    cout << "cout << int Point3D::* offsetx =  " << offsetx << endl;

    cout << "cout << int Point3D::* offsety =  " << offsety << endl;

    cout << "cout << int Point3D::* offsetz =  " << offsetz << endl;

输出结果:

int Point3D::* offsetx = 0

int Point3D::* offsety = 4

int Point3D::* offsetz = 8

cout << int Point3D::* offsetx =  1

cout << int Point3D::* offsety =  1

cout << int Point3D::* offsetz =  1

这里由于没有为Point3D定义<<操作,所以编译器这里自己偷偷的帮着你进了转化,输出结果就为1。用printf则可以直接将偏移量输出出来;

 

特别说明:

特别说明:

l        经过试验,在多重继承的情况下,使用第二种方式来获取Derived类的第二个基类中成员在Derived类对象中的偏移量,获得的结果却和直接对第二个基类进行取偏移量的结果相同,这一点的猜测是:多重继承中第二个(乃至后面)的成员的偏移量仍然按照基类中的布局显现——然而这明显和实际的内存布局情况不符,而且关于这点,还没有找到相关的说明,存疑?鉴于这种情况,推荐使用第一种方式进行操作;

l        在存在虚拟继承的情况下,计算来自虚基类成员偏移量时,对空对象成员取指针的操作会失败,crash产生;猜测原因是:现在的编译器都是通过在virtual function table中放置virtual base class的偏移量的方式来索引虚基类成员,因此如果使用空对象,那么所计算得到虚基类成员的方式就是 ((virtual base class *)(0+offset))-> member,这样得到的地址是不可访问的,从而导致程序崩溃;



这篇关于C++:获取类成员的偏移量的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

C#如何调用C++库

《C#如何调用C++库》:本文主要介绍C#如何调用C++库方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录方法一:使用P/Invoke1. 导出C++函数2. 定义P/Invoke签名3. 调用C++函数方法二:使用C++/CLI作为桥接1. 创建C++/CL

SpringBoot整合mybatisPlus实现批量插入并获取ID详解

《SpringBoot整合mybatisPlus实现批量插入并获取ID详解》这篇文章主要为大家详细介绍了SpringBoot如何整合mybatisPlus实现批量插入并获取ID,文中的示例代码讲解详细... 目录【1】saveBATch(一万条数据总耗时:2478ms)【2】集合方式foreach(一万条数

python获取网页表格的多种方法汇总

《python获取网页表格的多种方法汇总》我们在网页上看到很多的表格,如果要获取里面的数据或者转化成其他格式,就需要将表格获取下来并进行整理,在Python中,获取网页表格的方法有多种,下面就跟随小编... 目录1. 使用Pandas的read_html2. 使用BeautifulSoup和pandas3.

SpringBoot UserAgentUtils获取用户浏览器的用法

《SpringBootUserAgentUtils获取用户浏览器的用法》UserAgentUtils是于处理用户代理(User-Agent)字符串的工具类,一般用于解析和处理浏览器、操作系统以及设备... 目录介绍效果图依赖封装客户端工具封装IP工具实体类获取设备信息入库介绍UserAgentUtils

C++如何通过Qt反射机制实现数据类序列化

《C++如何通过Qt反射机制实现数据类序列化》在C++工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作,所以本文就来聊聊C++如何通过Qt反射机制实现数据类序列化吧... 目录设计预期设计思路代码实现使用方法在 C++ 工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作。由于数据类

C# foreach 循环中获取索引的实现方式

《C#foreach循环中获取索引的实现方式》:本文主要介绍C#foreach循环中获取索引的实现方式,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录一、手动维护索引变量二、LINQ Select + 元组解构三、扩展方法封装索引四、使用 for 循环替代

Linux下如何使用C++获取硬件信息

《Linux下如何使用C++获取硬件信息》这篇文章主要为大家详细介绍了如何使用C++实现获取CPU,主板,磁盘,BIOS信息等硬件信息,文中的示例代码讲解详细,感兴趣的小伙伴可以了解下... 目录方法获取CPU信息:读取"/proc/cpuinfo"文件获取磁盘信息:读取"/proc/diskstats"文

Vue3组件中getCurrentInstance()获取App实例,但是返回null的解决方案

《Vue3组件中getCurrentInstance()获取App实例,但是返回null的解决方案》:本文主要介绍Vue3组件中getCurrentInstance()获取App实例,但是返回nu... 目录vue3组件中getCurrentInstajavascriptnce()获取App实例,但是返回n

C++使用printf语句实现进制转换的示例代码

《C++使用printf语句实现进制转换的示例代码》在C语言中,printf函数可以直接实现部分进制转换功能,通过格式说明符(formatspecifier)快速输出不同进制的数值,下面给大家分享C+... 目录一、printf 原生支持的进制转换1. 十进制、八进制、十六进制转换2. 显示进制前缀3. 指

C++中初始化二维数组的几种常见方法

《C++中初始化二维数组的几种常见方法》本文详细介绍了在C++中初始化二维数组的不同方式,包括静态初始化、循环、全部为零、部分初始化、std::array和std::vector,以及std::vec... 目录1. 静态初始化2. 使用循环初始化3. 全部初始化为零4. 部分初始化5. 使用 std::a