C语言联合体:内存共享的妙用

联合体

联合体(Union)是C语言中一种特殊的用户自定义数据类型,它允许在同一内存位置存储不同的数据类型。与结构体不同,联合体的所有成员共享同一块内存空间,这意味着任何时候只能使用联合体的一个成员。

结构体(struct)中的每个成员都有自己的内存空间,而联合体(union)的所有成员共享同一块内存空间。这是两者最本质的区别。

cpp 复制代码
// 结构体示例
struct Example {
    int i;     // 4字节
    float f;   // 4字节
    char c;    // 1字节
}; // 总大小约为12字节(考虑内存对齐)

// 联合体示例
union Example {
    int i;     // 4字节
    float f;   // 4字节
    char c;    // 1字节
}; // 总大小为4字节(最大成员的大小)

联合体的大小等于其最大成员的大小 (可能会因内存对齐而增加)。所有成员的起始地址都相同,它们相互覆盖。

cpp 复制代码
union 联合体标签名 {
    成员类型1 成员名1;
    成员类型2 成员名2;
    // ...更多成员
};

声明

cpp 复制代码
union Data data;  // 声明一个Data类型的联合体变量

union Data {
    int i;
    float f;
    char str[20];
} data1, data2;

初始化

cpp 复制代码
union Data data = {10};  // 初始化第一个成员i为10

union Data data = {.f = 3.14};  // 初始化成员f为3.14

联合体成员的访问

使用点运算符访问
cpp 复制代码
#include <stdio.h>
#include <string.h>

int main() {
    union Data {
        int i;
        float f;
        char str[20];
    };
    
    union Data data;
    
    // 给联合体成员赋值
    data.i = 10;
    printf("data.i: %d\n", data.i);
    
    // 给另一个成员赋值(会覆盖之前的值)
    data.f = 220.5;
    printf("data.f: %.1f\n", data.f);
    // 此时data.i的值已经被覆盖,不再是10
    printf("data.i: %d\n", data.i);  // 输出一个"奇怪"的值
    
    // 给字符数组成员赋值
    strcpy(data.str, "C语言");
    printf("data.str: %s\n", data.str);
    // 此时data.i和data.f的值都被覆盖
    
    return 0;
}
使用指针访问
cpp 复制代码
#include <stdio.h>

int main() {
    union Data {
        int i;
        float f;
        char str[20];
    };
    
    union Data data;
    union Data *ptr = &data;  // 定义联合体指针并指向data
    
    // 通过指针给联合体成员赋值
    ptr->i = 10;
    printf("ptr->i: %d\n", ptr->i);
    
    // 通过指针给另一个成员赋值
    ptr->f = 220.5;
    printf("ptr->f: %.1f\n", ptr->f);
    
    return 0;
}

联合体内存访问

cpp 复制代码
#include <stdio.h>

int main() {
    union Example {
        char c;     // 1字节
        short s;    // 2字节
        int i;      // 4字节
        double d;   // 8字节
    };
    
    union Example ex;
    
    printf("联合体大小: %lu字节\n", sizeof(union Example));  // 通常输出8
    
    printf("c的地址: %p\n", &ex.c);
    printf("s的地址: %p\n", &ex.s);
    printf("i的地址: %p\n", &ex.i);
    printf("d的地址: %p\n", &ex.d);
    // 所有成员的地址都相同
    
    return 0;
}

注意:

使用联合体时,最重要的是要记住任何时候只有一个成员的值是有效的。给一个成员赋值会覆盖其他成员的值。

cpp 复制代码
union Data {
    int i;
    float f;
};

union Data data;
data.i = 10;
data.f = 3.14;  // 此时data.i的值被覆盖,不再是10
cpp 复制代码
union {
    int i;
    float f;
} data;

data.i = 123456;
printf("%f\n", data.f);  // 类型不同,不安全。输出一个"奇怪"的浮点数
相关推荐
little~钰1 小时前
可持久化线段树和标记永久化
算法
獭.獭.2 小时前
C++ -- 二叉搜索树
数据结构·c++·算法·二叉搜索树
TOYOAUTOMATON2 小时前
自动化工业夹爪
大数据·人工智能·算法·目标检测·机器人
im_AMBER2 小时前
Leetcode 67 长度为 K 子数组中的最大和 | 可获得的最大点数
数据结构·笔记·学习·算法·leetcode
feifeigo1233 小时前
MATLAB实现两组点云ICP配准
开发语言·算法·matlab
fengfuyao9853 小时前
粒子群算法(PSO)求解标准VRP问题的MATLAB实现
开发语言·算法·matlab
Ayanami_Reii3 小时前
进阶数据结构应用-SPOJ 3267 D-query
数据结构·算法·线段树·主席树·持久化线段树
guygg884 小时前
基于全变差的压缩感知视频图像重构算法
算法·重构·音视频
VT LI4 小时前
SDF在实时图形渲染中的核心原理与架构创新
算法·sdf·有号距离场