一、函数原型
#include <stddef.h>
size_t offsetof(type, member);
参数:
- type:某种类型(结构体等)
- member:type类型结构中某一成员
- 返回值:返回给定类型中给定成员的偏移量,以字节为单位
二、注意事项
- 这个宏很有用,因为组成结构的字段的大小在不同的实现中可能不同,编译器可能插入不同的数字。字段之间的填充字节数。因此,一个元素的偏移量不一定由前一个元素的大小之和给出。
- 如果成员未与字节边界对齐(即,它是位字段),则会导致编译器错误
宏定义的系统原型:
三、使用案例
- 我们知道域套接字地址的sun_path在不同的系统之间实现不同,所以想要获取域套接字地址的sun_path大小,offsetof宏是最好的选择,也增加了程序的可移植性
- 下面程序中先调用offsetof宏计算sum_path成员在socjaddr_un结构体中的偏移量,然后将结果与路径名长度相加。将得到的结果作为地址的总长度
#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <string.h>
#include <stddef.h>
#include <sys/un.h>
#include<sys/types.h>
#include<sys/socket.h>
int main()
{
int fd,size;
struct sockaddr_un un;
un.sun_family=AF_UNIX;
strcpy(un.sun_path,"foo.socket");
if((fd=socket(AF_UNIX,SOCK_STREAM,0))<0){
perror("socket");
exit(EXIT_FAILURE);
}
size=offsetof(struct sockaddr_un,sun_path)+strlen(un.sun_path);
if(bind(fd,(struct sockaddr*)&un,size)){
perror("bind");
exit(EXIT_FAILURE);
}
printf("UNIX domain socket bound\n");
exit(0);
}
- 第一次运行程序运行成功,并且创建了foo.socket文件
- 当第二次在运行程序时,bind检测到文件已经存在就出错退出
- 当我们删除foo.socket文件之后,再次运行成功