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

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

相关文章:

  • 如何通过OKR管理项目目标
  • 【第三十三周】BLIP论文阅读笔记
  • Flink02-学习-套接字分词
  • Ldap高效数据同步- MirrorMode双主复制模式配置详解(上)
  • 在 Cursor 中 配置 GitHub MCP Server
  • AI 应用同质化:一场看不见的资源 “吞噬战”
  • 软考:软件设计师考试数据结构知识点详解
  • HTML5 新特性详解:语义化标签、表单与音视频嵌入
  • 底层源码和具体测试解析HotSpot JVM的notify唤醒有序性(5000字详解)
  • JimuBI 积木报表 v1.9.5发布,大屏和仪表盘,免费数据可视化
  • 当AI浏览器和AI搜索替代掉传统搜索份额时,老牌的搜索引擎市场何去何从。
  • ubuntu 日志文件清空方式的解读
  • Ubuntu22.04/24.04 P104-100 安装驱动和 CUDA Toolkit
  • FFmpeg之三 录制音频并保存, API编解码从理论到实战
  • C++初阶-STL简介
  • Unity 和 Unreal Engine(UE) 两大主流游戏引擎的核心使用方法
  • 司法大模型构建指南
  • 模方ModelFun工程操作教程
  • Deep Dark Sea 局域網文件共享即時匿名聊天去數據庫部署
  • 1、Linux操作系统下,ubuntu22.04版本切换中英文界面
  • 俄外长与美国务卿通电话,讨论俄美关系及乌克兰问题
  • 一季度全国城镇新增就业308万人
  • 税务部门曝光3起通过拆分经营骗享小规模纳税人税费优惠偷税案件
  • 夜读丨修车与“不凑合”
  • 小米首次参加上海车展:没有雷军依旧人气爆棚,YU7上市时间未推迟
  • 宁德时代校友红利!副董事长给母校复旦豪捐10亿,曾毓群给交大捐近14亿