
在两篇博客《结构体详解:声明、使用与优化技巧》《结构体内存对齐》的基础上,咱们来看看结构体实现位段的能力。
位段的声明和结构是类似的,有两个不同:
int、unsigned int 或signed int 、char,即整形家族都是可以的。struct A
{
int _a : 2;
int _b : 5;
int _c : 10;
int _d : 30;
};A就是⼀个位段类型。 那咱们来试着分析一下,它的大小是多少:

可以用代码来测试一下:
#include<stdio.h>
struct A
{
//4
int _a : 2;
int _b : 5;
int _c : 10;
int _d : 30;
};
int main()
{
printf("%d\n", sizeof(struct A));
return 0;
}输出:

那咱们就详细看一下位段的内存分配吧
int unsigned int signed int 或者是 char等类型那咱们可以先来试着分析一下,空间是如何开辟的? 先看代码:
struct S
{
char a : 3;
char b : 4;
char c : 5;
char d : 4;
};
int main()
{
printf("%d", sizeof(struct S));
return 0;
}分析:

输出:

说明在vs2022咱们的假设是成立的。 接着看,在上面的基础上,对于主函数中的代码该怎么理解?
struct S
{
char a : 3;
char b : 4;
char c : 5;
char d : 4;
};
int main()
{
struct S s = { 0 };
s.a = 10;
s.b = 12;
s.c = 3;
s.d = 4;
return 0;
}对于 struct S s = { 0 };来说,内存中的变化看下图:

ok,那咱们来看后面部分: 咱们先要知道一个点位段中的成员在内存中从左向右分配,还是从右向左分配标准尚未定义。,咱们假设它是从右往左分配。



存储完毕,验证一下,首先转化为16进制:

在vs2022上面调试:

说明在vs2022上面是这样存储的。为什么要这样说呢,因为位段存在跨平台问题。咱们将展开来讲解:
总结: 跟结构相比,位段可以达到同样的效果,并且可以很好的节省空间,但是有跨平台的问题存在。
下图是网络协议中,IP数据报的格式,我们可以看到其中很多的属性只需要几个bit位就能描述,这里使用位段,能够实现想要的效果,也节省了空间,这样网络传输的数据报大小也会较小⼀些,对网络的畅通是有帮助的。

位段的几个成员共有同⼀个字节,这样有些成员的起始位置并不是某个字节的起始位置,那么这些位 置处是没有地址的。内存中每个字节分配⼀个地址,**⼀个字节内部的bit位是没有地址的。 ** 所以不能对位段的成员使用&操作符,这样就不能使用scanf直接给位段的成员输入值,只能是先输入 放在⼀个变量中,然后赋值给位段的成员。 代码示例:
struct A
{
int _a : 2;
int _b : 5;
int _c : 10;
int _d : 30;
};
int main()
{
struct A sa = { 0 };
scanf("%d", &sa._b);//这是错误的
return 0;
}调试一下:

那正确的该怎么写呢,前面提到了,赋值:
struct A
{
int _a : 2;
int _b : 5;
int _c : 10;
int _d : 30;
};
int main()
{
struct A sa = { 0 };
//scanf("%d", &sa._b);//这是错误的
//正确的示范
int b = 0;
scanf("%d", &b);
sa._b = b;
return 0;
}位段是 C 语言“贴近硬件”特性的典型体现,它牺牲可移植性换取极致性能。使用时需在“节省内存”和“代码稳定”间权衡,尤其在嵌入式与网络编程中,位段能发挥关键作用,但务必谨慎处理跨平台问题!