【C语言进阶篇】offsetof宏的介绍 及其实现

简介: 【C语言进阶篇】offsetof宏的介绍 及其实现

一、offsetof宏的介绍

offsetof宏的作用是获取结构体中某个成员相对于结构体起始地址的偏移量。通过计算成员在结构体中的位置,它提供了一种可移植的方法来确定偏移量。

头文件 :

offsetof宏的通用形式:
#define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)
参数说明:
  • TYPE: 表示结构体或联合体的类型名。当你需要计算某个结构体类型中某个成员的偏移量时,应在此处指定该结构体类型。
  • MEMBER: 表示结构体(或联合体)中需要计算偏移量的成员名称。这是你希望得知其在结构体内部位置的字段。
用法举例:
#include<stdio.h>
#include<stddef.h>
struct ST
{
  char a;
  int b;
  float c;
};
int main()
{
  printf("%zd\n", offsetof(struct ST, b));
  printf("%zd\n", offsetof(struct ST, c));
  return 0;
}

 

二、offsetof宏的实现

(这里用一些更简短的符号表示)

#define MY_offsetof(S,m) (size_t) &(((S*)0)->m)
  • S代表结构体的类型。
  • m代表结构体中的成员。
  • (S*)0这里将整数值0强制转换为指向结构体S类型的指针。这实际上创建了一个指向无效地址(NULL)的指针,但由于我们只是用它来计算偏移量,并不实际访问内存,因此这是安全的。
  • ->m使用->运算符来访问结构体指针的成员m。尽管指针是NULL,但这个操作在编译时仅用于计算偏移量,并不真正执行内存访问。
  • &(((S*)0)->m):取成员m的地址。由于我们有一个指向结构体类型的NULL指针,这个地址实际上就是成员m在结构体中的偏移量。
  • (size_t):将计算出的偏移量转换为size_t类型,这是一个无符号整数类型,通常用于表示对象的大小或内存中的偏移量。

MY_offsetof宏会返回结构体S中成员m的偏移量,这个偏移量是从结构体的起始地址到成员m的地址之间的距离(以字节为单位)。

#include<stdio.h>
#define MY_offsetof(S,m) (size_t)&(((S*)0)->m)
 
struct ST
{
  char a;
  int b;
  float c;
};
int main()
{
  printf("%zd\n", MY_offsetof(struct ST, b));
  printf("%zd\n", MY_offsetof(struct ST, c));
  return 0;
}

三、offsetof宏的使用注意事项

  • 仅适用于结构体和联合体:它不能用于普通变量、数组或非聚合类型(即不包含其他成员的简单类型)。
  • 编译时计算offsetof宏是在编译时计算偏移量的,因此它不能用于运行时动态生成的结构体类型或成员名。
  • 不涉及内存访问:虽然宏的表达式看起来像在访问内存,但实际上并没有发生真正的内存访问,因为所使用的指针指向的是未分配的地址。编译器仅依据类型信息计算偏移量,不会导致运行时错误。
相关文章
|
1月前
|
C语言
【C语言】全局搜索变量却找不到定义?原来是因为宏!
使用条件编译和 `extern` 来管理全局变量的定义和声明是一种有效的技术,但应谨慎使用。在可能的情况下,应该优先考虑使用局部变量、函数参数和返回值、静态变量或者更高级的封装技术(如结构体和类)来减少全局变量的使用。
39 5
|
5月前
|
编译器 程序员 C语言
C语言 宏
C语言 宏
51 5
|
6月前
|
Linux C语言
C语言宏IS_REACHABLE
C语言宏IS_REACHABLE
53 1
|
8月前
|
安全 编译器 程序员
C语言(16)----预处理中的宏以及预处理指令
C语言(16)----预处理中的宏以及预处理指令
75 2
|
7月前
|
存储 C语言
【C语言进阶篇】整数在内存的存储——原码、反码、补码
【C语言进阶篇】整数在内存的存储——原码、反码、补码
|
7月前
|
编译器 程序员 C语言
【C语言进阶篇】assert宏 使用详解
【C语言进阶篇】assert宏 使用详解
|
8月前
|
C语言
C语言进阶21收尾(编程练习)(atoi,strncpy,strncat,offsetof模拟实现+找单身狗+宏交换二进制奇偶位)(下)
C语言进阶21收尾(编程练习)(atoi,strncpy,strncat,offsetof模拟实现+找单身狗+宏交换二进制奇偶位)
48 0
|
8月前
|
C语言
C语言进阶21收尾(编程练习)(atoi,strncpy,strncat,offsetof模拟实现+找单身狗+宏交换二进制奇偶位)(上)
C语言进阶21收尾(编程练习)(atoi,strncpy,strncat,offsetof模拟实现+找单身狗+宏交换二进制奇偶位)
119 0
|
21小时前
|
算法 C语言
【C语言程序设计——函数】利用函数求解最大公约数和最小公倍数(头歌实践教学平台习题)【合集】
本文档介绍了如何编写两个子函数,分别求任意两个整数的最大公约数和最小公倍数。内容涵盖循环控制与跳转语句的使用、最大公约数的求法(包括辗转相除法和更相减损术),以及基于最大公约数求最小公倍数的方法。通过示例代码和测试说明,帮助读者理解和实现相关算法。最终提供了完整的通关代码及测试结果,确保编程任务的成功完成。
25 15
|
21小时前
|
C语言
【C语言程序设计——函数】亲密数判定(头歌实践教学平台习题)【合集】
本文介绍了通过编程实现打印3000以内的全部亲密数的任务。主要内容包括: 1. **任务描述**:实现函数打印3000以内的全部亲密数。 2. **相关知识**: - 循环控制和跳转语句(for、while循环,break、continue语句)的使用。 - 亲密数的概念及历史背景。 - 判断亲密数的方法:计算数A的因子和存于B,再计算B的因子和存于sum,最后比较sum与A是否相等。 3. **编程要求**:根据提示在指定区域内补充代码。 4. **测试说明**:平台对代码进行测试,预期输出如220和284是一组亲密数。 5. **通关代码**:提供了完整的C语言代码实现
33 24