结构体,枚举,联合

1.结构体

目录

1.匿名结构体

2.结构体自引用

3内存对齐

4.嵌套对齐

5.位段

2.枚举

3.联合

1. 1.匿名结构体 :就是不起名的意思,注意匿名结构体只能用一次。如果两个匿名结构体的成员类型一样在编译器看来也是不同类型。

 看见报错了吧,所以只能用一次,这不是重点,一般很少用。

2.结构体自引用:结构体中自己调用自己呗

 3.结构体内存对齐(重点知识)

内存对齐原因:

 1.平台原因:一些资料上是这样说的,“不是所有的硬件平台都能访问任意地址上的任意数据;某些硬件平台只能在某些特定地址处取某些特定的数据,否则就会抛出硬件异常”。也就是说在计算机在内存读取数据时,只能在规定的地址处读数据,而不是内存中任意地址都是可以读取的。

2.效率原因:正是由于只能在特定的地址处读取数据,所以在访问一些数据时,对于访问未对齐的内存,处理器需要进行两次访问;而对于对齐的内存,只需要访问一次就可以。其实这是一种以空间换时间的做法,但这种做法是值得的。

对齐规则:

1.第一个成员在与结构体偏移为0的地址处。

2.其他成员变量要对齐到对其数的整数倍地址处(对齐数:编译器默认的一个对齐数与该成员大小的较小值,vs的对齐数是8)。

3.结构体对齐数大小是最大对齐数的整数倍。

为啥这个地址开辟的是12字节的空间而不是1+1+4=6字节? 

看这个图,如果编译器开辟一块空间,那c1从0地址处开始一个字节占一个空间。而i是4字节 ,vs的默认对齐数是8,8>4,所以i的对齐数取最小的那个4,所以i要从4的倍数的地址编号处开始存放,所以编号是1,2,3的地址处被浪费了。而i是4字节从4-7都存的ta。同理,c2是1字节,1<8,

c2的对齐数是1,从8存一个空就可以了。但这就完了吗?c1,i,c2中对齐数最大的是4,而开辟空间要是这个最大对齐数的整数倍,但现在我们只用了9字节,所以要继续开辟空间到12,所以又往后开辟3字节,你学废了吗?各位看官是不是觉得这样太浪费空间了?对,没错。有啥改进办法没?肯定有啊,我把c1,c2这样的小字节的放一起,大字节的尽量放最后面。

这样是不是少浪费四个。分析一下,c1存0地址处,c2存1地址处,i继续存4的倍数4地址处,从4-7而8是最大对齐数4的倍数所以8字节空间足以,这就OK了,省4字节。

4.嵌套对齐:结构体中在嵌套一个结构体又如何对齐。

咱分析一下这个为啥是16.

s1是8字节的结构体,所以把s看做一个8字节变量,由上边的分析s从8地址处开始,8-15存放它,所以16字节空间就行。

如果结构体大小大于vs对齐数8呢?

分析一模一样。结构体s的对齐数是8,因为12>8,所以是8,s从12地址处开始存,占12字节到23地址处,所以24个字节空间就行。但是这种写法相对于上个浪费很多空间。

5.位段 (重点)

首先位段成员类型是int ,unsigned int,signed int。记牢。位段的写法也是有要求的。

位段只能写成int n:数字;这种类型。一字节是8比特,47比特,6字节不就行了,为啥8字节,博主这犊子骗我,诸位道友请和我一起诛杀博主,各位看官别急,我们在下面会分析的。

#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>


struct x
{
	char a : 3;
	char b : 4;
	char c : 5;
	char d : 4;
};
int main()
{
	struct x s = { 0 };
	printf("%d", sizeof(s));
	s.a = 10;//a是10二进制是1010
	s.b = 12;//1100
	s.c = 3;//0011
	s.d = 4;//0100
	return 0;
}

各位道友请看此码,请问占几字节?——2。3+4+5+4=16比特=2字节。

来编译一下

来人,把说2的叉出去 !

为啥是三

位段的跨平台问题:

 int 位段被当成有符号数还是无符号数是不确定的。
 位段中最大位的数目不能确定。(16位机器最大16,32位机器最大32,写成27,在16位机器会出问 题)。
 位段中的成员在内存中从左向右分配,还是从右向左来存储C语言中无具体标准。
 当一个结构包含两个位段,第二个位段成员比较大,无法容纳于第一个位段剩余的位时,是舍弃剩余的 位还是利用,这是不确定的。
2.枚举:枚举关键字“enum”

注意同类型的才能枚举,比如一周从周一到周日可以枚举,你不能周一,周二,白色,黑色,早餐这样干。

它是从0开始的连续的,那我不想叫他连续,我给YELLOW赋值它啥顺序呢?

这样式的。

3.联合体 

关键字:union

注意联合体变量成员共用一块空间,就比如这里应该5字节,内存对齐是8字节,但这是4字节,而a和x的地址一样。

其内存 最小也要能存下变量中最大那个字节数。

判断大小端存储

 在联合体中我们利用共同占一块空间也可以判断。

这个精妙于x和c的地址一样如果是小端则c直接返回数值,如果是大端和c无关,因为c存的是0,所以会返回0.

写了8个小时服了。 

  • 25
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 21
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 21
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值