一、奇妙的“结构体”

解析一下:上面是一个非常简单的代码,其中第一种结构体的初始化是linux源码中非常常见的一种方式,这种初始化的方式编译器必须要遵循ISO C99标准,否则只能使用第二种比较常规的方式。


(资料图片)

3、0地址与结构体的妙用

这一块的内容算是这篇文章的重点内容,各位小伙伴们要做好笔记了,作者一直非常强调一点的是多读读大佬们的代码,并且善于总结一些常用的小技巧供大家平时使用。这不,我们今天就拿Linux的kernal中的两个宏定义来分享几个结构体小技巧:

解析一下:第一个宏定义的功能是获得一个结构体成员距离结构体首地址的偏移量,参数TYPE : 结构体类型;参数MEMBER : 结构体成员,其实这个算是比较简单的,把0地址强制类型转化为结构体类型指针,然后通过结构体指向成员即可获得结构体成员变量,然后通过&进行取地址便获得了结构体成员地址,成员的偏移 = (结构体成员地址 - 结构体首地址);然而结构体首地址为0,这样成员的偏移 = 结构体成员地址,应该足够清楚了吧。

第二个宏定义的功能是通过结构体成员变量获得对应的结构体首地址(也就是结构体地址),参数ptr :结构体成员变量地址;参数type : 结构体类型;参数member :结构体成员,这个宏定义可能对于一些小伙伴而言在写法上有一点点难度,不过其主要分为两个部分:

第一部分通过typeof获得成员的类型并定义了一个const指针,定义为const的目的是不让用户对0地址的内容进行写操作,对于大部分芯片对不合法区域进行读写会引起异常;第二部分通过使用offsetof宏定义获得结构体成员相对结构体首地址的偏移,这样一相减便获得了当前结构体成员所属结构体的地址,原理公式:(结构体地址 = 结构体成员地址 - 结构体成员的偏移)。

这里大体说一下注意事项:在第二点我们谈到了typeof关键字,该关键字是GUN C标准中扩展的关键字,所以在使用该宏定义的时候需要注意一下;不然,采用其他标准进行编译可能会报错。

4、最后帮助大家理解的小程序

作者要说的全在代码里面了:

程序运行的结果如下:

推荐内容