Offsetof

#define offsetof(s, m)   (size_t)&(((s *)0)->m)

 

s是一个结构名,它有一个名为m的成员(sm 是宏offsetof的形参,它实际是返回结构s的成员m的偏移地址.

 

(s *)0 是骗编译器说有一个指向类(或结构)s的指针,其地址值

 

&((s *)0)->m   是要取得类s中成员变量m的地址. 因基址为0,这时m的地址当然就是ms中的偏移

 

最后转换size_t 型,即unsignedint

 

有例子如: 

struct  

    int  i; 

    int  j; 

}; 

   

struct   A   *pA; 

pA = new   A; 

这时,pA实际上是一个Pointer,指向某一确定的内存地址, 0x1234; 

pA->i 整体是一个int型变量,其地址是&(pA->i),'&'为取址运算符; 

那么&(pA->i)一定等于0x1234, i 是结构体A的第一个元素。 

&(pA->j)一定是0x1234 +0x4 = 0x1238; 因为sizeof(int) = 4; 

   

这个做法的巧妙之处就是:它把“0”作为上例中的pA,那么&(pA->j)就是 j offset 

 

解析结果是: 

(s*)0, 0 强制转换为Pointer to "s"    

可以记 pS = (s*)0pS是指向s的指针,它的值是0; 

那么pS->m就是m这个元素了,而&(pS->m)就是m的地址,就是offset

 

`memcpy()` 和 `offsetof()` 都是在 C 编程语言中使用的函数,分别用于数据复制和获取结构体成员偏移地址的功能。 ### memcpy() `memcpy()` 函数是一个库函数,通常包含于 `<string.h>` 头文件中。它用于将一个指定大小的数据块从源位置复制到目标位置。其语法形式大致如下: ```c void *memcpy(void *dest, const void *src, size_t n); ``` 其中, - `dest` 是指向目标存储区的指针; - `src` 是指向源存储区的指针; - `n` 是需要复制的字节数。 例如,在两个数组之间的复制操作可以像下面这样实现: ```c char str1[] = "Hello"; char str2; memcpy(str2, str1, sizeof(str1)); // 此时 str2 的内容会被设置为 'H', 'e', 'l', 'l', 'o' ``` ### offsetof() `offsetof()` 函数也是一个宏,同样位于 `<stddef.h>` 或 `<stdalign.h>` 头文件中,用于计算结构体成员相对于结构体起始处的偏移量。它的基本语法如下: ```c #define offsetof(struct_type, member_name) ((size_t)(intptr_t)&((struct_type *)0)->member_name) ``` 这里, - `struct_type` 是结构体的名称(实际上是指向该结构体类型的指针类型); - `member_name` 是想要获取偏移量的成员变量名。 例如,如果有一个结构体 `Person` 包含姓名、年龄等信息,你可以使用 `offsetof()` 来获取某个字段的位置: ```c struct Person { char name[20]; int age; }; // 获取名字字段相对于 Person 结构体的偏移量 size_t name_offset = offsetof(struct Person, name); ``` ### 相关问题: 1. `memcpy()` 和 `memmove()` 之间有何区别? 2. `offsetof()` 对于结构体和联合体有何影响? 3. 如何有效利用 `memcpy()` 和 `offsetof()` 提高程序性能?
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值