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++是两种广泛使用的编程语言,特别是在系统编程、嵌入式开发以及高性能计算领域,:本文主要介绍C/C++错误信息处理的常见方法及函数,文中通过代码介绍... 目录前言1. errno 和 perror()示例:2. strerror()示例:3. perror(

C++变换迭代器使用方法小结

《C++变换迭代器使用方法小结》本文主要介绍了C++变换迭代器使用方法小结,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录1、源码2、代码解析代码解析:transform_iterator1. transform_iterat

详解C++中类的大小决定因数

《详解C++中类的大小决定因数》类的大小受多个因素影响,主要包括成员变量、对齐方式、继承关系、虚函数表等,下面就来介绍一下,具有一定的参考价值,感兴趣的可以了解一下... 目录1. 非静态数据成员示例:2. 数据对齐(Padding)示例:3. 虚函数(vtable 指针)示例:4. 继承普通继承虚继承5.

C++中std::distance使用方法示例

《C++中std::distance使用方法示例》std::distance是C++标准库中的一个函数,用于计算两个迭代器之间的距离,本文主要介绍了C++中std::distance使用方法示例,具... 目录语法使用方式解释示例输出:其他说明:总结std::distance&n编程bsp;是 C++ 标准

Python获取中国节假日数据记录入JSON文件

《Python获取中国节假日数据记录入JSON文件》项目系统内置的日历应用为了提升用户体验,特别设置了在调休日期显示“休”的UI图标功能,那么问题是这些调休数据从哪里来呢?我尝试一种更为智能的方法:P... 目录节假日数据获取存入jsON文件节假日数据读取封装完整代码项目系统内置的日历应用为了提升用户体验,

微信公众号脚本-获取热搜自动新建草稿并发布文章

《微信公众号脚本-获取热搜自动新建草稿并发布文章》本来想写一个自动化发布微信公众号的小绿书的脚本,但是微信公众号官网没有小绿书的接口,那就写一个获取热搜微信普通文章的脚本吧,:本文主要介绍微信公众... 目录介绍思路前期准备环境要求获取接口token获取热搜获取热搜数据下载热搜图片给图片加上标题文字上传图片

C++ 中的 if-constexpr语法和作用

《C++中的if-constexpr语法和作用》if-constexpr语法是C++17引入的新语法特性,也被称为常量if表达式或静态if(staticif),:本文主要介绍C++中的if-c... 目录1 if-constexpr 语法1.1 基本语法1.2 扩展说明1.2.1 条件表达式1.2.2 fa

使用Python实现获取网页指定内容

《使用Python实现获取网页指定内容》在当今互联网时代,网页数据抓取是一项非常重要的技能,本文将带你从零开始学习如何使用Python获取网页中的指定内容,希望对大家有所帮助... 目录引言1. 网页抓取的基本概念2. python中的网页抓取库3. 安装必要的库4. 发送HTTP请求并获取网页内容5. 解

C++中::SHCreateDirectoryEx函数使用方法

《C++中::SHCreateDirectoryEx函数使用方法》::SHCreateDirectoryEx用于创建多级目录,类似于mkdir-p命令,本文主要介绍了C++中::SHCreateDir... 目录1. 函数原型与依赖项2. 基本使用示例示例 1:创建单层目录示例 2:创建多级目录3. 关键注

C++从序列容器中删除元素的四种方法

《C++从序列容器中删除元素的四种方法》删除元素的方法在序列容器和关联容器之间是非常不同的,在序列容器中,vector和string是最常用的,但这里也会介绍deque和list以供全面了解,尽管在一... 目录一、简介二、移除给定位置的元素三、移除与某个值相等的元素3.1、序列容器vector、deque