c语言分层理解(c语言结构体(下))

简介: 结构体之前有讲解结构体相关知识,这里对其进行补充说明一下要点。之前的文章请点击这里了解一下结构体。1.1 疑惑1.1.1 结构体类型是什么?

结构体

之前有讲解结构体相关知识,这里对其进行补充说明一下要点。之前的文章请点击这里了解一下结构体。

1.1 疑惑

1.1.1 结构体类型是什么?

6f7a857df1bed0ab187ae799e46dd51a.png

1.1.2 匿名结构体类型是什么?

e49b712571b17865fa61e723f8d6b0b2.png

1.1.3 typedef定义结构体类型问题

92d52b391008960abe2ff7b8019f99a0.png

1.1.4 初始化的两种方法理解

64c2c8e3b7aa05751c93a6b82b6d75c3.png

1.2 结构体内存对齐(计算结构体大小)

如何计算?

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

2.其他成员要对齐到某个对齐数的整数倍的偏移处(对齐数是什么?编译器默认的对齐数与该成员大小中两者的较小值(其中,VS的默认对齐数的值是8,Linux环境默认对齐数是该成员自身的大小))。

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

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

如何计算?看下面实例

1.2.1 实例一

struct S1
{
  char c1;
  int i;
  char c2;
};
int main()
{
  printf("%zd\n", sizeof(struct S1));
  return 0;
}
//最终输出是多少?

解释:

bb64e72bdf26cd921c9c47b80dc724c2.png

1.2.2 实例二

struct S2
{
  char c1;
  char c2;
  int i;
};
int main()
{
  printf("%zd\n", sizeof(struct S1));
  return 0;
}
//最终输出是多少?

解释:

b20f7e2d143e58cb63dfd939fa1e555b.png


上图中S1应该是S2。手误!

1.2.3 实例三

struct S2
{
  char c1;
  char c2;
  int i;
};
struct S3
{
  char c1;
  struct S2 s;
  double d;
};
int main()
{
  printf("%d\n", sizeof(struct S3));
  return 0;
}
//最终输出是多少?

解释:

fae662cbb8deab51a8851f5025434892.png

1.3 为什么存在内存对齐?

平台原因(移植原因):

不是所有的硬件平台都能访问任意地址上的任意数据的;某些硬件平台只能在某些地址处取某些特定类型的数据,否则抛出硬件异常。

2.性能原因:

数据结构(尤其是栈)应该尽可能地在自然边界上对齐。原因在于,为了访问未对齐的内存,处理器需要作两次内存访问;而对齐的内存访问仅需要一次访问。

总结:结构体的内存对齐都是拿空间来换时间

注意:设计结构体时,我们应该尽量让占用空间小的成员集中在一起,就像上述的实例二中的做法一样。

1.4 修改默认对齐数

这里用到#pragma预处理指令,如果想把对齐数改为4,就用这样的代码进行操作:#pragma pack(4) 使用完后要恢复操作:#pragma pack()

1.5 位段

1.5.1 什么是位段?

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

  1. 位段的成员必须是int、unsigned int、signed int。
  2. 位段的成员名后有一个冒号和一个数字(这个数字的单位是bit)

位段展示:

struct Bit
{
  int _a : 2;
  int _b : 5;
  int _c : 10;
  int _d : 30;
};

1.5.2 位段的内存分配

位段的成员可以是int、unsigned int、signed int、char类型。(必须是这四个类型)

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

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

实例:

struct S
{
  char a : 3;
  char b : 4;
  char c : 5;
  char d : 4;
};
int main()
{
  struct S s = { 0 };
  s.a = 10;
  s.b = 12;
  s.c = 3;
  s.d = 4;
  printf("%zd\n", sizeof(struct S));
  return 0;
}

解析:

136d2af0f21b2371ac6f7ac2b877aea9.png

这里是只是一种存储方式,存储方式不同,结果也会不同,这只是猜想的存储方式,正好满足visual studio 2022的结果。

1.5.3 位段的跨平台问题

1.int位段被当成有符号还是无符号不确定。

2.位段中最大位数不确定(16位机器下最大是16位,32位机器下是32位,这里如果在32位机器下分配空间>32的话是有问题的。

3.位段的成员在内存中从左向右分配还是从右向左分配是不确定的。

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

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

1.5.4 位段的应用

虽然位段使用的时候有很多不确定性,但是它可以节省空间成本。特定场景下还是可以用起来的,比如:我们使用手机发送消息时,对方接收到消息时,网络中就用到了相关位段的知识。


相关文章
|
1月前
|
存储 网络协议 编译器
【C语言】深入解析C语言结构体:定义、声明与高级应用实践
通过根据需求合理选择结构体定义和声明的放置位置,并灵活结合动态内存分配、内存优化和数据结构设计,可以显著提高代码的可维护性和运行效率。在实际开发中,建议遵循以下原则: - **模块化设计**:尽可能封装实现细节,减少模块间的耦合。 - **内存管理**:明确动态分配与释放的责任,防止资源泄漏。 - **优化顺序**:合理排列结构体成员以减少内存占用。
167 14
|
1月前
|
存储 编译器 C语言
【C语言】结构体详解 -《探索C语言的 “小宇宙” 》
结构体通过`struct`关键字定义。定义结构体时,需要指定结构体的名称以及结构体内部的成员变量。
197 10
|
2月前
|
存储 数据建模 程序员
C 语言结构体 —— 数据封装的利器
C语言结构体是一种用户自定义的数据类型,用于将不同类型的数据组合在一起,形成一个整体。它支持数据封装,便于管理和传递复杂数据,是程序设计中的重要工具。
|
2月前
|
存储 C语言
C语言如何使用结构体和指针来操作动态分配的内存
在C语言中,通过定义结构体并使用指向该结构体的指针,可以对动态分配的内存进行操作。首先利用 `malloc` 或 `calloc` 分配内存,然后通过指针访问和修改结构体成员,最后用 `free` 释放内存,实现资源的有效管理。
218 13
|
2月前
|
存储 编译器 数据处理
C 语言结构体与位域:高效数据组织与内存优化
C语言中的结构体与位域是实现高效数据组织和内存优化的重要工具。结构体允许将不同类型的数据组合成一个整体,而位域则进一步允许对结构体成员的位进行精细控制,以节省内存空间。两者结合使用,可在嵌入式系统等资源受限环境中发挥巨大作用。
87 11
|
2月前
|
存储 人工智能 算法
数据结构实验之C 语言的函数数组指针结构体知识
本实验旨在复习C语言中的函数、数组、指针、结构体与共用体等核心概念,并通过具体编程任务加深理解。任务包括输出100以内所有素数、逆序排列一维数组、查找二维数组中的鞍点、利用指针输出二维数组元素,以及使用结构体和共用体处理教师与学生信息。每个任务不仅强化了基本语法的应用,还涉及到了算法逻辑的设计与优化。实验结果显示,学生能够有效掌握并运用这些知识完成指定任务。
70 4
|
3月前
|
存储 C语言
如何在 C 语言中实现结构体的深拷贝
在C语言中实现结构体的深拷贝,需要手动分配内存并逐个复制成员变量,确保新结构体与原结构体完全独立,避免浅拷贝导致的数据共享问题。具体方法包括使用 `malloc` 分配内存和 `memcpy` 或手动赋值。
105 10
|
3月前
|
存储 大数据 编译器
C语言:结构体对齐规则
C语言中,结构体对齐规则是指编译器为了提高数据访问效率,会根据成员变量的类型对结构体中的成员进行内存对齐。通常遵循编译器默认的对齐方式或使用特定的对齐指令来优化结构体布局,以减少内存浪费并提升性能。
|
3月前
|
编译器 C语言
共用体和结构体在 C 语言中的优先级是怎样的
在C语言中,共用体(union)和结构体(struct)的优先级相同,它们都是用户自定义的数据类型,用于组合不同类型的数据。但是,共用体中的所有成员共享同一段内存,而结构体中的成员各自占用独立的内存空间。
|
3月前
|
存储 C语言
C语言:结构体与共用体的区别
C语言中,结构体(struct)和共用体(union)都用于组合不同类型的数据,但使用方式不同。结构体为每个成员分配独立的内存空间,而共用体的所有成员共享同一段内存,节省空间但需谨慎使用。

热门文章

最新文章