C语言-结构体-进阶小技巧

初始化有讲究

我们大部分初学的小伙伴可能都不怎么会跟结构体变量直接进行初始化,(哈哈,可能很多小伙伴定了了变量根本就不会进行初始化)不过还是要养成比较好的编程习惯,虽然现在大部分集成开发环境都会为大家把一些全局变量初始化为0,不过对于代码的可移植性、可预知性考虑还是建议既然定义了就要给一个初始状态。

 #include <stdio.h>
 //结构体定义
 typedef struct __tag_Test
 {
     int param1;
     int param2;
 }stTest;
 //结构体初始化方式1
 stTest sTest1 = {
    .param1 = 1,
    .param2 = 2
};
//结构体初始化方式2
stTest sTest2 = {
  1,2
};
/*********************************************
 * Fuction: main
 * Author : (公众号:最后一个bug)
 ********************************************/
int main(void) { 

    printf("sTest1.param1 = %d\n",sTest1.param1);
    printf("sTest1.param2 = %d\n",sTest1.param2);
    printf("sTest2.param1 = %d\n",sTest2.param1);
    printf("sTest2.param2 = %d\n",sTest2.param2);
    return 0;
}

解析一下:上面是一个非常简单的代码,其中第一种结构体的初始化是linux源码中非常常见的一种方式,这种初始化的方式编译器必须要遵循ISO C99标准,否则只能使用第二种比较常规的方式,不过现在大部分编译器都支持该标准所以也被比较普遍的使用,其带来的好处有几点:1)对于结构体成员变量的初始化更加清晰,特别是当结构体特别大的时候;2)不用在乎初始化的顺序,只需要名字和数据对应上即可,非常的灵活。

给同个结构体取多个名字

你进行C编程时间久了以后,对于所属性一致的变量等你都会用结构体进行封装,那么可能出现同一个结构体可能多个地方使用的情况,并且结构体的作用会有所不同,就好像一个人他可能是一名学生,也有可能在外面兼职做一名服务员等等,如果这个人不换一身衣服或者做个标记什么的可能有时候我们难以分辨,那么在C程序里面我们会怎样为他们换个衣服呢?简单代码如下:

#include <stdio.h>
 
  struct __tag_Man
 {
     int Age;
     int Height;
 };
 
 typedef struct __tag_Man stStudent;
typedef struct __tag_Man stWaiter;

stStudent sStudent = {
  .Age    = 12,
  .Height = 20,
};

stWaiter sWaiter = {
  .Age    = 12,
  .Height = 20,
};

int main(void) { 

    printf("sStudent.Age    = %d\n",sStudent.Age);
    printf("sStudent.Height = %d\n",sStudent.Height);
    printf("sWaiter.Age     = %d\n",sWaiter.Age);
    printf("sWaiter.Height  = %d\n",sWaiter.Height);
    printf("公众号:最后一个bug");
    return 0;
}

解析一下:上面的代码确实挺简单的,可能有些小伙伴一眼就可以看懂,不过这种使用方法时非常有价值的,特别是以后大家玩算法,比如说:节点都是一样的,不过节点分为子节点和父节点,对于一些处理函数传入的参数虽然是一样的,不过意义却不同,我们就可以通过这样的方式进行处理,从而增加代码可读性

0地址与结构体的妙用

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

#define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER) 
#define container_of(ptr, type, member) ({ \
         const typeof( ((type *)0)->member ) *__mptr = (ptr); \
         (type *)( (char *)__mptr - offsetof(type,member) );})

解析一下:

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

2)第二个宏定义的功能是通过结构体成员变量获得对应的结构体首地址(也就是结构体地址),参数ptr :结构体成员变量地址;参数type : 结构体类型;参数member :结构体成员,这个宏定义可能对于一些小伙伴而言在写法上有一点点难度,不过其主要分两部分,第一部分通过typeof获得成员的类型并定义了一个const指针,定义为const的目的是不让用户对0地址的内容进行写操作,对于大部分芯片对不合法区域进行读写会引起异常。第二部分通过使用offsetof宏定义获得结构体成员相对结构体首地址的偏移,这样一相减便获得了当前结构体成员所属结构体的地址,原理公式:(结构体地址 = 结构体成员地址 - 结构体成员的偏移)

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

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

#include <stdio.h> 

/**********************************
 * Fuction : from Linux Kernal 
 **********************************/
#define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER) 
#define container_of(ptr, type, member) ({ \
               const typeof( ((type *)0)->member ) *__mptr = (ptr); \
               (type *)( (char *)__mptr - offsetof(type,member) );})

/**********************************
 * Fuction : 结构体定义区
 * Author  : (公众号:最后一个bug)
 **********************************/
typedef struct _tag_Test{ 
    int Member1; 
    int Member2; 
    int Member3; 
}STest;

int main(void) { 
    int iTest = 0;
    STest stTest;

    printf("offsetof(STest , Member3) : %d\n",offsetof(STest , Member3));  
    printf("&stTest                   : 0X%X\n",&stTest);    
    printf("&(stTest.Member3)         : 0X%X\n",&(stTest.Member3));  
    printf("container of Member3      : 0X%X\n",container_of(&(stTest.Member3),STest,Member3));

    //这里主要是进一步让大家理解第二个宏
    //且右侧必须加()
    iTest = ({int Val = 5; Val;});

    printf("iTest = %d\n",iTest);
    printf("欢迎关注公众号:最后一个bug\n");
    return 1; 
}
程序运行的结果如下:
offsetof(STest , Member3) : 8
&stTest                   : 0X452469E0
&(stTest.Member3)         : 0X452469E8
container of Member3      : 0X452469E0
iTest = 5

参考

【典藏】大佬们都在用的结构体进阶小技巧

  • 28
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值