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个小时服了。