C/C++ 位段

目录

什么是位段?

位段的内存分配

位段的跨平台问题


什么是位段?

位段的声明与结构是类似的,但是有两个不同:

  • 位段的成员必须是 int、unsigned int 或signed int 等整型家族。
  • 位段的成员名后边有一个冒号和一个数字

这是一个结构体

struct S
{
	int a;
	int b;
	int c;
	int d;
};

这是一个位段

struct A
{
	int a : 2;
	int b : 5;
	int c : 10;
	int d : 30;
};

位段的样子看起来奇奇怪怪的,那个冒号和后面的数字是什么意思呢?

位段位段,它名字里的“位”就是二进制位。

冒号和后面的数字其实是它的成员变量在告诉编译器:

  • 成员a:我只需要2个比特的空间足以!
  • 成员b:请给我分配5个比特的空间吧!
  • 成员c:给我来10个!
  • 成员d:我需要的空间大,给我来40个比特的空间吧!

每个成员都得到了自己想要的大小的空间,那么位段A的总大小是多少呢?

我们用sizeof来计算一下:

#include

struct S
{
	int a;
	int b;
	int c;
	int d;
};

struct A
{
	int a : 2;
	int b : 5;
	int c : 10;
	int d : 30;
};

int main()
{
	printf("结构体S的大小为:  %d字节\n", sizeof(struct S));
	printf("位段A的大小为:     %d字节\n", sizeof(struct A));

	return 0;
}

学完上一章之后,我们很轻松的计算出结构体S的大小4*4=16字节与结果一致。

再看位段,把成员们所需要的空间大小加起来2+5+10+30=47比特。

已知1字节=8比特。换算一下位段A的总大小应该是6字节就够了,可结果却是8字节。

位段的内存分配

位段的成员可以是 int unsigned int signed int 或者是 char (属于 整形家族 )类型。

位段的空间上是按照需要以4个字节( int )或者1个字节( char )的方式来开辟的。

位段涉及很多不确定因素,位段是不跨平台的,注重 可移植的程序应该避免使用位段 。

首先声明:位段有很多用法是标准未定义的,也就是说不同的机器上或者不同的编译器上内存如何

分配都有所差异,所以一下内容中有关内存分配的插图并不是严格正确的!!!

举个例子:

这是一个成员是int类型的位段:

struct A
{
	int a : 2;
	int b : 5;
	int c : 10;
	int d : 30;
};

C/C++ 位段_第1张图片

这是一个成员是char类型的位段:

struct S
{
	char a : 3;
	char b : 4;
	char c : 5;
	char d : 4;
};

C/C++ 位段_第2张图片

现在假定义一个位段的变量并进行初始化:

struct S
{
	char a : 3;
	char b : 4;
	char c : 5;
	char d : 4;
};
struct S s = { 0 };
s.a = 10;
s.b = 12;
s.c = 3;
s.d = 4;

其在VS2013环境下是这样进行存储数据的:

C/C++ 位段_第3张图片

说明数据是以二进制的方式进行存储的

再次强调:以上的存储形式仅仅是vs2013环境下的,每种不同的编译器在不同的环境下都有自己存储的方式 。

位段的跨平台问题

问题一:int 位段被当成有符号数还是无符号数是不确定的。

说明:例如一个整型存储的时候,它的最高位是有符号位还是无符号位是有明确的规定的,而位段标准并没有规定。有的平台会当作有符号数处理,有的反之。

问题二:位段中最大位的数目不能确定。

说明:16位机器最大16,32位机器最大32,写成27,在16位机器会出问题。

问题三:位段中的成员在内存中从左向右分配,还是从右向左分配标准尚未定义。

说明:

C/C++ 位段_第4张图片

问题四:当一个结构包含两个位段,第二个位段成员比较大,无法容纳于第一个位段剩余的位时,是舍弃剩余的位还是利用,这是不确定的。

说明:

C/C++ 位段_第5张图片

总结:跟结构相比,位段可以达到同样的效果,但是可以很好的节省空间,但是有跨平台的问题存在。

你可能感兴趣的:(C/C++,c语言)