C语言结构体内存对齐

简介: 结构体类型的声明结构体自引用结构体变量的定义和初始化结构体内存对齐结构体传参

结构体

结构体的声明

结构体是一些值集合的,里面可以包括char,int,double等等的各种类型构成的一个新的类型,也就是说相当于自己基于C语言本身给出的类型自己把它组成一个集合。

它的声明格式如下:


struct 类型名{
  元素1;
  元素2;
  .....
};
``
例如描述一个学生
```cpp
struct  stu
{
  char name[20];//名字
  int age;//年龄
  char ID[20];//学号
};

结构体的自引用

我们知道结构体里面可以包括char,int ,double等等类型,那么通过上面我们讲的结构体它也是一种类型。只不过是我们自己定义的一种类型,那么我们可以在结构体里面定义一个结构体吗?


struct  stu
{
  char name[20];//名字
  int age;//年龄
  char ID[20];//学号
  struct stu next;
};

这里我们可以知道直接定义是不行的。

那么正确的自引用方式是下面这个

struct  stu
{
  char name[20];//名字
  int age;//年龄
  char ID[20];//学号
  struct stu *next;
};

这里在C语言后期链表期间给大家详细讲解


结构体变量的定义和初始化

struct  stu
{
  char name[20];//名字
  int age;//年龄
  char ID[20];//学号
}s1,s2;     //直接在结构体类型声明后直接定义
struct stu s1, s2;    //定义结构体
  struct stu s3 = { "张三",18,2022001 };//定义后直接初始化
  struct nau
  {
    int num;//号数
    char name[20];//名字
  } a1={18,"李四"};//直接在结构体类型声明后直接定义并初始化
  struct cmp
  {
    int i;
    struct nau;//结构体嵌套初始化
  }p1 = { 1,{18,"丽子"} };

结构体内存对齐

基本了解结构体的内容之后,我们就要计算结构体的大小了。

这也是一个特别热门的考点:

那么我们直接上几道练习


#include<stdio.h>
struct s
{
  char i;
  char j;
  int a;
}s1;
struct c
{
  char b;
  int b1;
  char b2;
}s2;
int main()
{
  printf("%d\n", sizeof(s1));//大小是多少呢? 
  printf("%d\n", sizeof(s2));//大小是多少呢?
  return 0;
}

如何计算?

首先得掌握结构体的对齐规则:

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

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

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

VS中默认的值为8

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

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


为什么存在内存对齐?

平台原因(移植原因):

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

性能原因:

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

简单来说就是用空间换取时间

967d9274076447fba4d24c16027141bf.png

那在设计结构体的时候,我们既要满足对齐,又要节省空间,我们要 让占用空间小的成员尽量集中在一起

struct S3
{
 double d;  //练习1
 char c;
 int i;
};
printf("%d\n", sizeof(struct S3));
//练习2-结构体嵌套问题
struct S4
{
 char c1;
 struct S3 s3;
 double d;
};
printf("%d\n", sizeof(struct S4));

上面s3的大小是16

s4的大小是32


结构体传参

详细请见结构体详解

对于结构体详解,后期会出改良优化过后的版本


相关文章
|
16天前
|
存储 搜索推荐 算法
【数据结构】树型结构详解 + 堆的实现(c语言)(附源码)
本文介绍了树和二叉树的基本概念及结构,重点讲解了堆这一重要的数据结构。堆是一种特殊的完全二叉树,常用于实现优先队列和高效的排序算法(如堆排序)。文章详细描述了堆的性质、存储方式及其实现方法,包括插入、删除和取堆顶数据等操作的具体实现。通过这些内容,读者可以全面了解堆的原理和应用。
59 16
|
25天前
|
存储 C语言
如何在 C 语言中实现结构体的深拷贝
在C语言中实现结构体的深拷贝,需要手动分配内存并逐个复制成员变量,确保新结构体与原结构体完全独立,避免浅拷贝导致的数据共享问题。具体方法包括使用 `malloc` 分配内存和 `memcpy` 或手动赋值。
31 10
|
24天前
|
存储 大数据 编译器
C语言:结构体对齐规则
C语言中,结构体对齐规则是指编译器为了提高数据访问效率,会根据成员变量的类型对结构体中的成员进行内存对齐。通常遵循编译器默认的对齐方式或使用特定的对齐指令来优化结构体布局,以减少内存浪费并提升性能。
|
29天前
|
编译器 C语言
共用体和结构体在 C 语言中的优先级是怎样的
在C语言中,共用体(union)和结构体(struct)的优先级相同,它们都是用户自定义的数据类型,用于组合不同类型的数据。但是,共用体中的所有成员共享同一段内存,而结构体中的成员各自占用独立的内存空间。
|
29天前
|
存储 C语言
C语言:结构体与共用体的区别
C语言中,结构体(struct)和共用体(union)都用于组合不同类型的数据,但使用方式不同。结构体为每个成员分配独立的内存空间,而共用体的所有成员共享同一段内存,节省空间但需谨慎使用。
|
1月前
|
编译器 C语言 C++
C语言结构体
C语言结构体
25 5
|
1月前
|
编译器 Linux C语言
C语言 之 结构体超详细总结
C语言 之 结构体超详细总结
20 0
|
1月前
|
编译器 C语言 Python
C语言结构
C语言结构
16 0
|
1月前
|
存储 编译器 Linux
深入C语言:探索结构体的奥秘
深入C语言:探索结构体的奥秘
|
1月前
|
存储 编译器 C语言
c语言回顾-结构体(2)(下)
c语言回顾-结构体(2)(下)
30 0