C 语言编程教程

C 语言 - 首页

C 语言基础

C 语言 - 概述 C 语言 - 特性 C 语言 - 发展历史 C 语言 - 环境设置 C 语言 - 程序结构 C 语言 - Hello World C - 编译过程 C - 注释 C - 标记 C - 关键字 C - 标识符 C - 用户输入 C - 基本语法 C - 数据类型 C - 变量 C - 整数提升 C - 类型转换 C - 类型转换 C - 布尔值

C 语言中的常量和文字

C - 常量 C - 字面量 C - 转义序列 C - 格式说明符

C 语言中的运算符

C - 运算符 C - 算术运算符 C - 关系运算符 C - 逻辑运算符 C - 位运算符 C - 赋值运算符 C - 一元运算符 C - 递增和递减运算符 C - 三元运算符 C - sizeof 运算符 C - 运算符优先级 C - 其他运算符

C 语言中的决策

C - 决策 C - if 语句 C - if...else 语句 C - 嵌套 if 语句 C - switch 语句 C - 嵌套 switch 语句

C 语言中的循环

C - 循环 C - While 循环 C - For 循环 C - Do...while 循环 C - 嵌套循环 C - 无限循环 C - Break 语句 C - Continue 语句 C - goto 语句

C 语言中的函数

C - 函数 C - Main 函数 C - 按值调用函数 C - 按引用调用函数 C - 嵌套函数 C - 可变参数函数 C - 用户定义函数 C - 回调函数 C - return 语句 C - 递归

C 语言中的作用域规则

C - 作用域规则 C - 静态变量 C - 全局变量

C 语言中的数组

C - 数组 C - 数组的属性 C - 多维数组 C - 将数组传递给函数 C - 从函数返回数组 C - 可变长度数组

C 语言中的指针

C - 指针 C - 指针和数组 C - 指针的应用 C - 指针运算 C - 指针数组 C - 指向指针的指针 C - 将指针传递给函数 C - 从函数返回指针 C - 函数指针 C - 指向数组的指针 C - 指向结构体的指针 C - 指针链 C - 指针 vs 数组 C - 字符指针和函数 C - NULL 指针 C - void 指针 C - 悬垂指针 C - 解引用指针 C - Near、Far 和 Huge 指针 C - 指针数组的初始化 C - 指针与多维数组

C 语言中的字符串

C - 字符串 C - 字符串数组 C - 特殊字符

C 语言的结构体和联合

C - 结构体 C - 结构体和函数 C - 结构体数组 C - 自引用结构 C - 查找表 C - 点 (.) 运算符 C - 枚举(或 enum) C - 结构填充和打包 C - 嵌套结构 C - 匿名结构和联合 C - 联合 C - Bit 位字段 C - Typedef

C 语言中的文件处理

C - 输入和输出 C - 文件 I/O(文件处理)

C 语言中的预处理器

C - 预处理器 C - #pragma 编译指示 C - 预处理器操作符 C - 宏 C - 头文件

C 语言中的内存管理

C - 内存管理 C - 内存地址 C - 存储类

C 其他主题

C - 错误处理 C - 可变参数 C - 命令执行 C - 数学函数 C - static 静态关键字 C - 随机数生成 C - 命令行参数

C 语言编程资源

C语言问题与解答答案 C语言快速指南 C语言速查表 C语言实用资源 C语言讨论


C 语言中的结构填充和打包

C 语言中的结构填充是什么?

C 语言中的结构填充是由 CPU 架构 处理的过程。结构填充会在结构中添加一定数量的空字节,以使数据成员在内存中自然对齐。对齐要求由处理器架构而非语言本身决定。当然,对齐要求会根据数据总线大小或特定 CPU 架构的其他架构考虑而变化。

通过示例理解结构体填充

我们定义一个结构体类型,如下所示 -

struct struct1 {
    char x;
    int y;
    char z;
};

示例 1

我们检查一下此类型变量所需的字节数 -

#include <stdio.h>

struct struct1{
   char a;
   char b;
   int c;
};

int main(){

   printf("Size: %d", sizeof(struct struct1));  
   return 0;
}

输出

运行此代码,将产生以下输出 -

Size:8

结果与预期相反。

考虑到 char 类型需要 1 个字节,而 int 类型需要 4 个字节,因此可能认为输出应为"1 + 1 + 4 = 6 个字节"。

结构填充

然而,CPU 架构需要改变此结构。考虑到我们使用的是 32 位 CPU,它一次读取 4 个字节,这意味着 1 个字等于 4 个字节。

在一个 CPU 周期内,它访问字符"a",然后是字符"b",以及 int"c"的前两个字节。在第二个周期内,它访问另外两个字节。

即使我们只想读取"c",也需要两个 CPU 周期。为此,CPU 会在存储"c"值的字节前添加两个空字节。这种机制称为填充

结构填充

这解释了我们上面得到的结果,即结构体类型的大小为 8 个字节。

示例 2

让我们更改上述结构体类型中成员的顺序,并设置"b"的类型和"c"的类型。

#include <stdio.h>

struct struct1{
   char a;
   int b;
   char c;
};

int main(){

   printf("size: %d", sizeof(struct struct1));
   return 0;
}

输出

运行代码并检查其输出 −

size: 12

在第一个字的 4 个字节中,第一个字节分配给了字符"a",后面跟着三个空字节。

构成下一个字的接下来 4 个字节用于存储 int"b"。随后,在接下来 4 个字节中,只有一个字节用于"c"。但是,结构体的大小为 12。

什么是 C 语言中的结构体打包?

另一方面,结构体打包是一种最小化填充影响的机制,从而尝试减少浪费的内存空间。我们可以使用某些指令和属性来实现打包。

通过示例理解结构打包

CPU 架构强制的填充是不可避免的,但是有一些方法可以最大限度地减少填充。可以使用 -

  • 使用 #pragma pack(1) 指令
  • 使用 packed 属性

使用 #pragma pack(1) 指令

#pragma pack(1) 预处理指令强制编译器忽略填充,并在内存分配过程中将结构成员首尾对齐。

示例

让我们将此指令添加到之前使用的代码顶部,并查看结果 -

#include <stdio.h>
#pragma pack(1)

struct struct1{
   char a;
   int b;
   char c;
};

int main(){

   printf("size: %d", sizeof(struct struct1));
   return 0;
}
输出

运行代码并检查其输出 −

size: 6

我们可以看到,结构体填充已被避免,并减少了内存浪费。

使用 __attribute__((packed))

在 GCC 中,我们可以使用属性来指定结构体和联合类型的各种特殊属性。这些属性包括:aligned、deprecated、packed、transparent_union、unusedvisibility。应用这些属性的语法为"__attribute__ ((...))"。

示例

在这里,我们将在结构体类型的定义中使用 packed 属性。

#include <stdio.h>

struct __attribute__((packed)) struct1{
   char a;
   int b;
   char c;
};

int main(){

   printf("size: %d", sizeof(struct struct1));
   return 0;
}
输出

运行代码并检查其输出 -

size: 6

此方法还可以避免填充的影响。