CONTAINING_RECORD宏原理与使用详解

2024-06-18 05:32

本文主要是介绍CONTAINING_RECORD宏原理与使用详解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

先不急着说CONTAINING_RECORD宏,我们从最浅显的代码开始讲解。

一、0指针的妙用

0指针,即nullptr、NULL,空指针,是不是很常见,一遇到它往往就是segment fault。代码,如下:

struct Test
{int a;float b;
};Test* pTest = nullptr;
int a_value = pTest->a; // segment fault
int b_value = pTest->b; // segment fault
pTest->a = 1;           // segment fault
pTest->b = 1.0;         // segment fault

上述想必没有疑问吧。空指针难道就没有用了?不,下面看一个巧妙的应用。

typedef unsigned long long quint64;
Test* pTest = nullptr;
quint64 offset_a = (quint64)(&(pTest->a));  // offset_a==0
quint64 offset_b = (quint64)(&(pTest->b));  // offset_b==4

offset_a为0,offset_b为4,如果你觉得很惊喜,请继续看。

之所以pTest->a,pTest->b没有报错,是因为前面添加了取地址符&,相当于我们告诉编译器,并不是真的要取a、b的值,我们只是取a、b的地址,然鹅Test对象的地址pTest为0,根据C++对象内存布局,如下图,可知当Test对象地址为0时,实际上取到的a、b地址,也就是a、b变量相对于整个对象首地址的偏移量。

在这里插入图片描述

小结:

利用某结构的空指针,对该结构成员取地址,可以得到该成员的偏移量。

换言之,我们很容易利用成员地址反推出整个对象的地址。

二、CONTAINING_RECORD宏

CONTAINING_RECORD宏定义位于winnt.h中,如下:

//
// Calculate the address of the base of the structure given its type, and an
// address of a field within the structure.
//#define CONTAINING_RECORD(address, type, field) ((type *)( \(PCHAR)(address) - \(ULONG_PTR)(&((type *)0)->field)))

该宏的功能,是根据某个结构体中成员变量的地址,计算出结构体地址。

  • address,成员变量地址
  • type,结构体类型
  • field,成员变量名

该宏定义的原理,就是上一章中介绍的使用0指针获取成员偏移,然后再使用成员变量地址-成员偏移,就得到了结构体地址。

该宏定义的使用,如下:

struct School
{int level;float cost;
};struct Student
{int age;School school;
};School sch;
sch.level = 1;
sch.cost = 5000;Student zhangsan;
zhangsan.age = 10;
zhangsan.school = sch;Student* pStu = CONTAINING_RECORD(&zhangsan.school, Student, school);
qDebug() << &zhangsan;
qDebug() << pStu;

运行结果:

在这里插入图片描述

pStu与&zhangsan值相等。

貌似看起来,好像这么操作一遍,没什么用处。但是实际上,在某些情况下,还是非常有用的一个宏。

三、使用C++封装使用CONTAINING_RECORD宏

代码如下:

#include <QCoreApplication>
#include <QDebug>//
// Calculate the address of the base of the structure given its type, and an
// address of a field within the structure.
//#define CONTAINING_RECORD(address, type, field) ((type *)( \(qint8*)(address) - \(quint64)(&((type *)0)->field)))
// 轮子
class Wheel
{
public:Wheel(int count, int color): count(count),color(color){}private:int count; // 数量int color; // 颜色
};// 汽车
class Car
{
public:Car(int seat, Wheel wheel): seat(seat),wheel(wheel){}Wheel* getWheel(){return &wheel;}// 将轮子转换为对应的汽车static Car *wheelToCar(Wheel *pWheel){return CONTAINING_RECORD(pWheel, Car, wheel);}private:int seat; // 座位数量Wheel wheel; // 轮子
};int main(int argc, char *argv[])
{QCoreApplication a(argc, argv);Wheel wheel(4, 1); // 4个轮子,1种颜色Car myCar(5, wheel); // 5个座位,1套轮子Car* pCar = Car::wheelToCar(myCar.getWheel());qDebug() << &myCar;qDebug() << pCar;return a.exec();
}

运行结果:

在这里插入图片描述

用途:

当我们需要将一个class中,某个成员对象(struct或class),取出来,传递给其他api,其他api将这个成员对象作为参数进行一些处理,处理完后,还会将该对象传递回给我们,这时候,我们希望能够通过此对象获取到所在的整个对象,那么我们使用这个C++封装的宏,实现起来会很方便。

例如:

win下的IOCP异步IO中,会把OVERLAPPED结构传给api,处理完,会传回OVERLAPPED结构体。

以及linux下libaio异步IO中,也会iocb结构传给api,处理完,会传回iocb结构体。



若对你有帮助,欢迎点赞、收藏、评论,你的支持就是我的最大动力!!!

同时,阿超为大家准备了丰富的学习资料,欢迎关注公众号“超哥学编程”,即可领取。

本文涉及工程代码,公众号回复:19ContainingRecord,即可下载。

在这里插入图片描述

这篇关于CONTAINING_RECORD宏原理与使用详解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

MySQL 8 中的一个强大功能 JSON_TABLE示例详解

《MySQL8中的一个强大功能JSON_TABLE示例详解》JSON_TABLE是MySQL8中引入的一个强大功能,它允许用户将JSON数据转换为关系表格式,从而可以更方便地在SQL查询中处理J... 目录基本语法示例示例查询解释应用场景不适用场景1. ‌jsON 数据结构过于复杂或动态变化‌2. ‌性能要

postgresql使用UUID函数的方法

《postgresql使用UUID函数的方法》本文给大家介绍postgresql使用UUID函数的方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录PostgreSQL有两种生成uuid的方法。可以先通过sql查看是否已安装扩展函数,和可以安装的扩展函数

Python实现终端清屏的几种方式详解

《Python实现终端清屏的几种方式详解》在使用Python进行终端交互式编程时,我们经常需要清空当前终端屏幕的内容,本文为大家整理了几种常见的实现方法,有需要的小伙伴可以参考下... 目录方法一:使用 `os` 模块调用系统命令方法二:使用 `subprocess` 模块执行命令方法三:打印多个换行符模拟

MySQL字符串常用函数详解

《MySQL字符串常用函数详解》本文给大家介绍MySQL字符串常用函数,本文结合实例代码给大家介绍的非常详细,对大家学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录mysql字符串常用函数一、获取二、大小写转换三、拼接四、截取五、比较、反转、替换六、去空白、填充MySQL字符串常用函数一、

Java中Arrays类和Collections类常用方法示例详解

《Java中Arrays类和Collections类常用方法示例详解》本文总结了Java中Arrays和Collections类的常用方法,涵盖数组填充、排序、搜索、复制、列表转换等操作,帮助开发者高... 目录Arrays.fill()相关用法Arrays.toString()Arrays.sort()A

如何使用Lombok进行spring 注入

《如何使用Lombok进行spring注入》本文介绍如何用Lombok简化Spring注入,推荐优先使用setter注入,通过注解自动生成getter/setter及构造器,减少冗余代码,提升开发效... Lombok为了开发环境简化代码,好处不用多说。spring 注入方式为2种,构造器注入和setter

MySQL中比较运算符的具体使用

《MySQL中比较运算符的具体使用》本文介绍了SQL中常用的符号类型和非符号类型运算符,符号类型运算符包括等于(=)、安全等于(=)、不等于(/!=)、大小比较(,=,,=)等,感兴趣的可以了解一下... 目录符号类型运算符1. 等于运算符=2. 安全等于运算符<=>3. 不等于运算符<>或!=4. 小于运

使用zip4j实现Java中的ZIP文件加密压缩的操作方法

《使用zip4j实现Java中的ZIP文件加密压缩的操作方法》本文介绍如何通过Maven集成zip4j1.3.2库创建带密码保护的ZIP文件,涵盖依赖配置、代码示例及加密原理,确保数据安全性,感兴趣的... 目录1. zip4j库介绍和版本1.1 zip4j库概述1.2 zip4j的版本演变1.3 zip4

Python 字典 (Dictionary)使用详解

《Python字典(Dictionary)使用详解》字典是python中最重要,最常用的数据结构之一,它提供了高效的键值对存储和查找能力,:本文主要介绍Python字典(Dictionary)... 目录字典1.基本特性2.创建字典3.访问元素4.修改字典5.删除元素6.字典遍历7.字典的高级特性默认字典

使用Python构建一个高效的日志处理系统

《使用Python构建一个高效的日志处理系统》这篇文章主要为大家详细讲解了如何使用Python开发一个专业的日志分析工具,能够自动化处理、分析和可视化各类日志文件,大幅提升运维效率,需要的可以了解下... 目录环境准备工具功能概述完整代码实现代码深度解析1. 类设计与初始化2. 日志解析核心逻辑3. 文件处