当前位置: 首页 > ai >正文

2025.4.27_C_Struct,Enum,Union

1.结构体Struct

1.1 概念

      一系列具有相同类型或不同类型的数据构成的数据集合。

struct person
{char* name;uint8_t gender;
}p1,p2;         //p1,p2是struct person类型的变量

 

1.2 匿名结构体类型

struct
{char* name;uint8_t gender;
};

这时结构体对象只能在声明类型的时候创建变量,其余时候因为没有名字,所以不能创建变量。

1.3 结构体的自引用

结构体的成员不能是自身,因为无法确定大小,但是可以是自身类型的指针(链表,树)。

1.4 typedef + 结构体

typedef struct person
{char* name;uint8_t gender;
}person;

        这种写法定义了一个struct person类型,并且给他定义了一个别名person,在创建对象的时候既可以使用struct person也可以直接使用person。 

typedef struct 
{char* name;uint8_t gender;
}person;

        这种写法定义了一个匿名结构体类型,并且给他指定了别名person,在创建对象时只能使用person,不能使用struct person,因为匿名结构体对象除了在声明的时候创建对象外其余时候都不能创建对象。 

1.5 结构体所占内存大小

        1.5.1结构体的对齐规则:

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

                2.其他成员变量要对齐到某个数字(对齐数)的整数倍的地址处。

                        对齐数=编译器默认的一个对齐数 与 该成员大小的较小值。

                3.结构体总大小为最大对齐数(每个成员变量都有一个对齐数)的整数倍。

                4.如果嵌套了结构体的情况,嵌套的结构体对齐到自己的最大对齐数的整数倍处,结构体的整体大小就是所有最大对齐数(含嵌套结构体的对齐数)的整数倍

例子1:
struct s2
{char c1;char c2;int i;
};
printf("%d\n", sizeof(struct s2)); // 输出8

        为什么会输出8?因为在按照规则2,存储b的时候他们需要对齐到char类型大小的的整数倍地址处,也就是要对齐1的整数倍,所以a存储在偏移量为0的地址,b存储在偏移量为1的地址;但是int c,按照规则,它需要对齐到4的整数倍的地址处,所以它的地址偏移量为4,中间偏移量为2和3的地址为空闲状态。所以这个结构体的整体大小为8。

 例子2:
struct s1
{char c1;int i;char c2;
};
printf("%d\n", sizeof(struct s1)); // 输出12

        为什么输出12?因为i是int类型,它的地址需要对齐4的整数倍,所以它的地址偏移量为4,c2只需要对齐1的整数倍就可以,所以目前占用地址数是9,但是根据 

规则3:结构体总大小为最大对齐数(每个成员变量都有一个对齐数)的整数倍

所以总大小需要对齐结构体成员中最大对齐数的整数倍(这里是4),但是9不满足,所以必须对齐到12.

 

例子3:
struct s3
{double d;char c;int i;
};
printf("%d\n", sizeof(struct s3)); // 输出16

        为什么输出16?因为成员d占用八个地址,c只需要对齐1的整数倍,所以它在偏移量为8的地址处,i需要对齐4的整数倍,明显9不满足,所以它存储在偏移量为12的地址处。所以目前s3占用16个字节,根据规则三:要对其成员中最大大小的整数倍,这里刚好是16,所以满足。

 

例子4:
struct s4
{char c1;struct s3 s;double d;
};
printf("%d\n", sizeof(struct s4)); // 输出32

        为什么输出32?c1在偏移量为0的地址处,s占用16个字节,但是根据规则2:

                2.其他成员变量要对齐到某个数字(对齐数)的整数倍的地址处。

                        对齐数=编译器默认的一个对齐数 与 该成员大小的较小值。

 其中,VS的默认对齐数是8,所以s只需要对齐偏移量为8的地址处,所以它在8-24的地址处;d是double类型,它需要对齐8的整数倍偏移量,刚好24是8的倍数,不用空闲地址,所以目前一共占用32个字节,根据规则4:

    4.如果嵌套了结构体的情况,嵌套的结构体对齐到自己的最大对齐数的整数倍处,结构体的整体大小就是所有最大对齐数(含嵌套结构体的对齐数)的整数倍

这里因为存储s的时候对齐的是VS默认的对齐数8,所以这里struct s4 占用的总空间只需要对齐8的倍数,而不是16。刚好32满足条件,所以struct s4的大小是32.

1.5.2 为什么存在内存对齐?

1.5.3 怎么修改编译器的默认字节对齐数?

#program pack(n)
...
#program pack()
 例子:
#pragma pack(1)
struct s3
{char c1;char c2;int i;
};
#pragma pack()

现在结构体s3中的编译器默认字节对齐数变成了1, 所以i只需要对齐1的整数倍,整个s3也只需要对齐1的整数倍偏移量,所以s3占用的字节数是6。

1.5.4 offsetof() 查看成员的结构体偏移量

struct s4
{char c1;struct s3 s;double d;
};int main()
{printf("%d", offsetof(struct s4 , d)); //输出16
}

2.枚举Enum

3.联合体Union

http://www.xdnf.cn/news/2354.html

相关文章:

  • 单片机学习笔记9.数码管
  • Redis使用总结
  • 相机DreamCamera2录像模式适配尺寸
  • 使用c++实现一个简易的量子计算,并向外提供服务
  • 一文说清Token这个大模型中的数字乐高积木的作用
  • MIT6.S081 - Lab10 mmap(文件内存映射)
  • 内耗型选手如何能做到不内耗?
  • MySQL最新安装、连接、卸载教程(Windows下)
  • Linux进程学习【环境变量】进程优先级
  • T8332FN凯钰LED驱动芯片多拓扑车规级AEC-Q100
  • 秒杀压测计划 + Kafka 分区设计参考
  • IP地址与子网计算工具
  • 0302洛必达法则-微分中值定理与导数的应用.md
  • 云原生课程-Docker
  • openstack创建虚拟机
  • 什么是模块化区块链?Polkadot 架构解析
  • 在Linux中,使用标准IO库,进行格式化IO操作
  • 深度解析Zemax优化函数:让光学设计从“能用”到“极致”的核心密码
  • 驱动开发硬核特训 · Day 22(下篇): # 深入理解 Power-domain 框架:概念、功能与完整代码剖析
  • I-CON: A Unifying Framework for Representation Learning
  • qt 3d航迹图
  • Scala集合操作与WordCount案例实战总结
  • Linux高效IO
  • SQL面试之--明明建了索引为什么失效了?
  • docker部署ruoyi系统
  • Rule.resourceQuery(通过路径参数指定loader匹配规则)
  • 【音视频】FFmpeg过滤器框架分析
  • django.db.models.query_utils.DeferredAttribute object
  • PDF嵌入图片
  • python连接Elasticsearch并完成增删改查