C语言内存管理——内存对齐与共用体union

C语言内存管理------内存对齐与共用体union

前言

在C语言中,内存管理是程序员必须掌握的核心技能。本文将深入探讨两个重要概念:内存对齐共用体(union)。理解它们不仅能帮你写出更高效的代码,还能在面试中脱颖而出。


一、内存对齐

1.1 什么是内存对齐

不管是结构体,还是普通的变量,都存在内存对齐的现象。简单来说,内存对齐就是数据只能存放在自己类型整数倍的内存地址上。

对齐规则: 数据只能放在自己类型整数倍的内存地址上。简单理解:内存地址 ÷ 占用字节 = 可以整除。

1.2 常见类型的对齐规则

数据类型 占用字节 对齐要求
char 1字节 内存地址能被1整除(任意位置)
short 2字节 内存地址能被2整除
int / float 4字节 内存地址能被4整除
long long / double 8字节 内存地址能被8整除

1.3 结构体的内存对齐

结构体的内存对齐在上面的基础上又多了一条规则:

⚠️ 结构体总大小规则: 结构体的总大小必须是其内部最大类型的整数倍(用来确定最后一个数据补位的情况)。

来看一个例子:

c 复制代码
struct Example {
    char a;     // 1字节
    int b;      // 4字节
    char c;     // 1字节
};

// 内存布局(假设起始地址为0):
// 地址0: a (char, 1字节)
// 地址1-3: 补3个空字节(int需要4字节对齐)
// 地址4-7: b (int, 4字节)
// 地址8: c (char, 1字节)
// 地址9-11: 补3个空字节(总大小需是int的整数倍,即4的倍数)
// 总大小:12字节

注意: 对齐的时候会补空白字节,但是不会改变原本字节的大小。char补位之后,本身还是1个字节。

1.4 优化技巧

为了节约空间,我们通常会把小的数据类型写在最上面,大的数据类型写在最下面。

c 复制代码
// 优化前:12字节
struct Bad {
    char a;     // 1字节
    int b;      // 4字节
    char c;     // 1字节
};

// 优化后:8字节
struct Good {
    char a;     // 1字节
    char c;     // 1字节(小的放一起)
    int b;      // 4字节
};

只是调整了成员顺序,就从12字节减少到了8字节,节省了33%的空间!


二、共用体union

2.1 什么是共用体

共用体,也叫联合体、共同体,是一种特殊的数据类型。它的特点是:所有成员共享同一块内存空间

2.2 共用体的特点

  1. 所有的变量都使用同一个内存空间
  2. 每次只能给一个变量进行赋值,因为第二次赋值时会覆盖原有的数据
  3. 所占的内存大小 = 最大成员的长度(也受内存对齐影响)
c 复制代码
#include <stdio.h>

union Data {
    int i;
    float f;
    char c;
};

int main() {
    union Data data;

    data.i = 10;
    printf("i = %d\n", data.i);  // i = 10

    data.f = 3.14f;
    printf("f = %.2f\n", data.f); // f = 3.14
    printf("i = %d\n", data.i);   // i的值被覆盖了,变成垃圾值

    printf("sizeof(union Data) = %zu\n", sizeof(union Data)); // 4字节
    return 0;
}

2.3 共用体的内存大小

  • 以最大的单个成员的长度为准
  • 总大小一定是最大单个成员的整数倍
  • 同样受内存对齐规则的影响

三、结构体和共用体的区别

对比项 结构体 struct 共用体 union
含义 一种事物中包含多个属性 一个属性有多种类型
存储方式 各存各的,互不影响 存一起,多次存会覆盖
内存占用 各个变量的总和(受内存对齐影响) 最大类型的大小(受内存对齐影响)
使用场景 描述一个对象的多个属性 节省内存、类型转换、协议解析

四、总结

  • 内存对齐:是CPU访问效率的优化机制,记住"小的放前面"的优化原则
  • 共用体:所有成员共享内存,适合"同一时间只用一种类型"的场景
  • 结构体 vs 共用体:结构体是"与"的关系,共用体是"或"的关系

掌握这些内存管理的细节,能让你对C语言的底层机制有更深刻的理解。

相关推荐
GV191rLvq1 小时前
基于Socket实现的最简单的Web服务器【ASP.NET原理分析】
服务器·前端·asp.net
charlie1145141911 小时前
Cinux: 加载第一个内核:从 bootloader 跳进 C++
linux·开发语言·c++·嵌入式
运维管理2 小时前
H3C SecPath W2000-G[AK]系列Web应用防火墙 典型配置举例(E6711 E6712 E6713)-6W108-H3C 官方配置
服务器·网络·php
C+-C资深大佬2 小时前
python while循环
服务器·开发语言·python
Tian_Hang2 小时前
eclipse ditto 学习笔记
运维·服务器·开发语言·javascript·3d
iCxhust2 小时前
linux目录是否保存在硬盘 启动后读入解析的
linux·运维·服务器
懒鸟一枚3 小时前
Linux 系统 Service 服务配置详解
linux·服务器·网络
敖行客 Allthinker3 小时前
企业级多台服务器组装 K3s 高性能集群实战指南
运维·服务器·团队开发
RisunJan3 小时前
Linux命令-readonly(Bash 内建设置只读变量)
linux