typeof、offsetof、container_of

 

typeof

用于获取一个对象的类型,比如:

unsigned int a = 1; // typeof (a) is unsigned int 
short b = 2;        // typeof (b) is short

  

offsetof

#define offsetof(TYPE, MEMBER)  ((size_t)&((TYPE *)0)->MEMBER)

获取某个成员变量(MEMBER)在其所在的结构体(TYPE)里的偏移:

  1. 首先将地址0强行转换为 TYPE * 类型
  2. 再将此对象中的MEMBER取地址后,强制转化为size_t格式返回

因为该对象的起始地址为0,那么MEMBER处的地址自然是其在此结构体中的偏移地址。

 

container_of

// non ISO variant from linux kernel; checks ptr type, but triggers 'ISO C forbids braced-groups within expressions [-Wpedantic]'
//  __extension__ is here to disable this warning
#define container_of(ptr, type, member)  ( __extension__ ({     \
        const typeof( ((type *)0)->member ) *__mptr = (ptr);    \
        (type *)( (char *)__mptr - offsetof(type,member) );}))

 

它的作用是根据一个结构体成员的地址来获取该结构体的地址:

  1. typeof( ((type *)0)->member ) 就是获取type->member的数据类型,并用该类型创建一个const 的指针  __mptr 
  2. __mptr的地址是传入的数据结构体成员的地址
  3. 把__mptr的地址强制转为(char *)的地址,然后再减去它在结构体中的偏移地址,不就是此结构体的地址了嘛
  4. 最后将此地址再强制转换为它本来的类型:type

 

posted @ 2020-04-02 14:10  QIYUEXIN  阅读(273)  评论(0编辑  收藏  举报