当前位置: 首页 > news >正文

C/C++获取结构体成员的偏移量

目录

1.通过构造结构体对象计算

2.offsetof宏获取

2.1.介绍

2.2.原理

2.3.注意事项


1.通过构造结构体对象计算

        通过创建结构体对象,获取结构体起始地址以及结构体成员的地址,然后将成员地址减去结构体起始地址,就可以得到该成员相对于结构体起始位置的偏移量。

#include <iostream>

// 定义一个结构体
struct MyStruct {
    char first;
    int second;
    double third;
};

int main() {
    // 创建一个结构体对象
    MyStruct obj;

    // 获取结构体起始地址
    char* structStart = reinterpret_cast<char*>(&obj);

    // 获取各成员的地址
    char* firstAddr = reinterpret_cast<char*>(&obj.first);
    char* secondAddr = reinterpret_cast<char*>(&obj.second);
    char* thirdAddr = reinterpret_cast<char*>(&obj.third);

    // 计算偏移量
    size_t offsetFirst = firstAddr - structStart;
    size_t offsetSecond = secondAddr - structStart;
    size_t offsetThird = thirdAddr - structStart;

    // 输出偏移量
    std::cout << "Offset of 'first': " << offsetFirst << " bytes" << std::endl;
    std::cout << "Offset of 'second': " << offsetSecond << " bytes" << std::endl;
    std::cout << "Offset of 'third': " << offsetThird << " bytes" << std::endl;

    return 0;
}

方法步骤: 

1)结构体定义:定义了一个名为 MyStruct 的结构体,包含 char 类型的 first、int 类型的 second 和 double 类型的 third 三个成员。
2)创建结构体对象:在 main 函数中创建了一个 MyStruct 类型的对象 obj。
3)获取地址:
使用 reinterpret_cast 将结构体对象的地址转换为 char* 类型,得到结构体的起始地址 structStart。
同样使用 reinterpret_cast 获取各成员的地址,并将其转换为 char* 类型。
4)计算偏移量:通过成员地址减去结构体起始地址,得到各成员的偏移量。
5)输出结果:将计算得到的偏移量输出到控制台。

2.offsetof宏获取

2.1.介绍

        在 C++ 中,你可以使用标准库中的 offsetof 宏来获取结构体中某个成员变量相对于结构体起始地址的偏移量。offsetof 宏定义在 <cstddef> 头文件中,它接受两个参数:结构体类型和结构体成员的名称,返回该成员相对于结构体起始位置的字节偏移量。

        示例代码:

#include <iostream>
#include <cstddef>

// 定义一个结构体
struct ExampleStruct {
    char a;
    int b;
    double c;
};

int main() {
    // 获取结构体成员的偏移量
    std::cout << "Offset of 'a' in ExampleStruct: " << offsetof(ExampleStruct, a) << " bytes" << std::endl;
    std::cout << "Offset of 'b' in ExampleStruct: " << offsetof(ExampleStruct, b) << " bytes" << std::endl;
    std::cout << "Offset of 'c' in ExampleStruct: " << offsetof(ExampleStruct, c) << " bytes" << std::endl;

    return 0;
}

方法步骤: 

1)结构体定义:定义了一个名为 ExampleStruct 的结构体,包含三个成员变量:char 类型的 a、int 类型的 b 和 double 类型的 c。
2)offsetof 宏的使用:在 main 函数中,使用 offsetof 宏分别计算 ExampleStruct 结构体中成员变量 a、b 和 c 相对于结构体起始地址的偏移量,并将结果输出到控制台。
3)输出结果:程序将输出每个成员变量的偏移量(以字节为单位)。

2.2.原理

vs2019的源码实现如下:

#if defined _MSC_VER && !defined _CRT_USE_BUILTIN_OFFSETOF
    #ifdef __cplusplus
        #define offsetof(s,m) ((::size_t)&reinterpret_cast<char const volatile&>((((s*)0)->m)))
    #else
        #define offsetof(s,m) ((size_t)&(((s*)0)->m))
    #endif
#else
    #define offsetof(s,m) __builtin_offsetof(s,m)
#endif
  • s:表示结构体或标准布局类的类型名。
  • m:表示结构体或标准布局类中成员的名称。

实现分析:

1)((s*)0)
把整数 0 强制转换为指向类型 s 的指针。这意味着创建了一个指向地址 0 处的 s 类型对象的指针。这里只是在语法层面模拟,并不会真正访问地址 0 处的内存。
2)((s*)0)->m
通过前面得到的指针来访问 s 类型对象的成员 m。同样,这只是一种语法操作,没有实际的内存访问。
3)reinterpret_cast<char const volatile&>((((s*)0)->m))
reinterpret_cast 是 C++ 中的一种强制类型转换运算符,这里将成员 m 转换为 char const volatile& 类型的引用。
char 类型的大小是 1 字节,将成员转换为 char 类型的引用有助于后续计算偏移量。
const 表示该引用指向的对象是常量,不能被修改。
volatile 关键字告诉编译器该对象的值可能会以编译器无法预知的方式被改变,防止编译器进行过度的优化。
4)&reinterpret_cast<char const volatile&>((((s*)0)->m))
对转换后的引用取地址。由于前面的指针是从地址 0 开始的,所以得到的地址值实际上就是成员 m 相对于结构体起始位置的偏移量。
5)(::size_t)
使用 ::size_t 进行强制类型转换。:: 是作用域解析运算符,这里使用全局作用域的 size_t 类型。size_t 是一种无符号整数类型,通常用于表示对象的大小或内存地址的偏移量,这样能确保偏移量以合适的整数类型表示。

2.3.注意事项

  • 标准布局要求:这个宏只能用于标准布局的结构体和类。如果是非标准布局的类型(如包含虚函数的类),由于内存布局的复杂性,该宏可能无法正确计算偏移量。
  • 可移植性:虽然这种实现方式在大多数编译器和平台上都能正常工作,但在一些特殊的嵌入式系统或编译器中,可能会因为地址 0 有特殊用途而导致问题。不过,标准库提供的 offsetof 是经过严格测试和优化的,具有更好的可移植性。
  • const 和 volatile 的使用:在这个宏中使用 const 和 volatile 主要是为了确保类型转换的正确性和防止编译器过度优化,但在实际计算偏移量时,通常不会影响最终结果。

总的来说,这种自定义 offsetof 宏的实现方式利用了指针运算和地址计算的特性,巧妙地计算出结构体成员的偏移量。

相关文章:

  • 【CXX】5.2 extern “C++“
  • 4.2 使用说明:手册写作利器VNote的使用
  • 大白话html第十一章
  • I²C总线应用场景及1.8V与3.3V电压选择
  • Nano-GraphRAG复现——只使用Ollama,无需API Key
  • 质量属性场景描述
  • IO基础练习4
  • CogToolBlock和CogIDTool工具
  • ES时序数据库的性能优化
  • C++ Primer 拷贝、赋值与销毁
  • 如何改变怂怂懦弱的气质(2)
  • 记录一次利用条件索引优化接口性能的实践
  • golang并发编程如何学习
  • unsloth-llama3-8b.py 中文备注版
  • 汽车零部件厂如何选择最适合的安灯系统解决方案
  • ESLint 深度解析:原理、规则与插件开发实践
  • C# Unity 面向对象补全计划 之 索引器与迭代器
  • Spring AI 1.0.0-M6 快速开始(一)
  • MySQL批量生成建表语句
  • 解决CentOS 8.5被恶意扫描的问题
  • 玉林一河段出现十年最大洪水,一村民被冲走遇难
  • 纽约市长称墨海军帆船撞桥已致2人死亡,撞桥前船只疑似失去动力
  • 上海公办小学验证今起开始,下周一和周二分区进行民办摇号
  • 美联储主席:供应冲击或更频繁,将重新评估货币政策方法中的通胀和就业因素
  • 证监会:2024年依法从严查办证券期货违法案件739件,作出处罚决定592件、同比增10%
  • 【社论】公平有序竞争,外卖行业才能多赢