#pragma pack() 易忽略的问题

#pragma pack() 易忽略的问题

  C/C++中,class、struct、union默认的大小对齐方式为按照成员变量所占空间最大的大小进行对齐,如:

typedef struct test{
    char a;
    int b;
    double c;
}test_t;

则test结构体内所有成员变量按照double类型进行字节对齐,此时sizeof(test_t)等于8*3。在网络通信中,为了让数据包更紧凑些,我们往往希望能够通过#pragma pack预编译命令改变默认对齐方式,#pragma pack()使用方式如下:
  #pragma pack (n)      作用:C编译器将按照n个字节对齐。
  #pragma pack ()        作用:取消自定义字节对齐方式。
  #pragma pack (push,1)     作用:是指把原来对齐方式设置压栈,并设新的对齐方式设置为一个字节对齐
  #pragma pack(pop)      作用:恢复对齐状态
  比如上面的test结构体:

#pragma pack(push, 1)
typedef struct test{
    char a;
    int b;
    double c;
}test_t;
#pragma pack(pop)

此时test结构体的对齐方式为单字节对齐,sizeof(test_t)=1+4+8,若存在多个#pragma pack (n),遵从向上对齐原则,即某个结构体定义上方最近的一个#pragma pack()。这里有一点需要特别注意,那就是#pragma pack()的作用域问题:
  如果未对#pragma pack进行取消,那么其作用域便是本文件,如果是头文件,则会影响到包含它的模块,如此一来,由于没有及时地设回去,会间歇性地莫名其妙地引起程序崩溃,就很讨厌,这种bug不注意是根本排查不出来的。所以在通过#pragma pack()设置了对齐方式后,一定要再将其设回去 .加入push和pop可以使对齐恢复到原来状态,而不是编译器默认,可以说后者更优,但是很多时候两者差别不大。

————————————————
版权声明:本文为CSDN博主「MCNS_37」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/qq_35308053/article/details/118296616

猜你喜欢

转载自blog.csdn.net/modi000/article/details/125045954
今日推荐