0

0

C++中联合体的大小是如何由其最大的成员决定的

P粉602998670

P粉602998670

发布时间:2025-09-11 11:41:01

|

412人浏览过

|

来源于php中文网

原创

联合体大小由其最大成员决定,因其所有成员共享同一内存区域,必须足够容纳最大成员并满足最严格对齐要求。结构体成员独立存储,大小为各成员之和加填充;联合体成员共享起始地址,任一时刻仅一个成员有效,大小至少为最大成员大小且为最严格对齐要求的倍数。典型应用包括节省内存(如消息负载)、实现变体类型(配合枚举)以及底层类型转换(现代C++推荐std::bit_cast替代)。

c++中联合体的大小是如何由其最大的成员决定的

C++联合体(union)的大小确实是由其所有成员中占用内存空间最大的那个成员决定的。这不是一个巧合,而是其设计哲学——节省内存——的必然结果。所有成员都共享同一块内存区域,因此这块区域必须足够大以容纳其中最大的“住客”。

解决方案

联合体的本质在于其内存共享机制。与结构体(struct)不同,结构体的成员是依次排列的,每个成员都有自己独立的内存空间,所以结构体的大小是所有成员大小之和(加上可能的填充字节)。而联合体的所有成员都从同一个内存地址开始存储。这意味着在任何给定时间,联合体只能存储其中一个成员的值。为了确保任何成员都能被完整地存储进去,编译器自然会分配一块足够大的内存区域,其大小至少要等于它所有成员中最大的那个。如果最大的成员是

double
(8字节),那么即使联合体里只有一个
char
(1字节)或
int
(4字节),整个联合体的大小也必须是8字节,因为
double
需要这8字节。

union MyUnion {
    char c;       // 1 byte
    int i;        // 4 bytes
    double d;     // 8 bytes
};
// 在大多数系统上,sizeof(MyUnion) 会是 8(或更多,由于对齐,但至少是8)。

这里其实隐含了一个重要的点,就是内存对齐。联合体的大小不仅要能容纳最大的成员,还要满足所有成员的对齐要求,特别是最大成员的对齐要求。

联合体与结构体在内存布局上的根本区别是什么?

结构体(struct)和联合体(union)在C++中都是复合数据类型,但它们在内存布局和使用方式上有着天壤之别。

立即学习C++免费学习笔记(深入)”;

  • 结构体 (struct): 它的成员是各自独立、顺序存储的。这意味着每个成员都拥有自己独立的内存地址,即使因为内存对齐规则,成员之间可能存在一些填充(padding)字节,它们也绝不会重叠。

    sizeof(struct)
    通常是所有成员大小之和,再加上编译器为了优化访问速度和满足对齐要求而可能插入的填充字节。你可以同时访问或修改结构体中的所有成员,它们的值互不影响。这很符合我们对“一篮子”不同物品的直观理解。

  • 联合体 (union): 它的成员共享同一块内存区域,从同一个起始地址开始存储。这意味着在任何给定时刻,你只能有效地使用其中一个成员。当你给一个成员赋值后,这块共享内存区域的内容就会被这个新值覆盖,其他成员的值可能变得不可预测或无意义(如果它们类型不同)。

    sizeof(union)
    是其所有成员中最大成员的大小,并同时考虑了内存对齐的要求。我个人在初学C++时,对联合体的这种“共享”概念感到有些困惑,总觉得它像一个多面手,但又不能同时施展所有技能。后来才明白,这正是它在嵌入式系统或需要极致内存优化的场景下大放异彩的原因。它不是为了方便同时处理多种数据类型,而是为了在不同时刻用同一块内存来表示不同类型的数据。

内存对齐如何影响联合体的实际大小?

即使联合体最大的成员是X字节,但实际的

sizeof(union)
可能比X大。这是因为内存对齐的要求。每个数据类型都有一个默认的对齐要求(例如,
int
通常要求4字节对齐,
double
要求8字节对齐)。处理器在访问内存时,通常会以字长(word size)的倍数来读取数据,如果数据没有对齐到合适的地址,处理器可能需要进行多次内存访问,或者引发性能下降,甚至在某些体系结构上直接报错。

联合体作为一个整体,其起始地址必须满足其所有成员中对齐要求最严格的那个。同时,联合体的总大小也必须是其最严格对齐要求的倍数,以确保在数组中,下一个联合体实例也能正确对齐。

WOBIZ电子商务2.0程序
WOBIZ电子商务2.0程序

WO@BIZ电子商务2.0软件是窝窝团队基于对互联网发展和业务深入研究后,采用互联网2.0的思想设计、开发的电子商务和社会化网络(SNS)结合的解决方案产品。WOBIZ是互联网2.0创业、传统网站转型、中小企业宣传产品网应用的最佳选择。 它精心设计的架构、强大的功能机制、友好的用户体验和灵活的管理系统,适合从个人到企业各方面应用的要求,为您提供一个安全、稳定、高效、 易用而快捷的电子商务2.0网络

下载

示例:

union AlignedUnion {
    char a;   // 1 byte, 默认对齐1字节
    int b;    // 4 bytes, 默认对齐4字节
    double c; // 8 bytes, 默认对齐8字节
};
// 在大多数64位系统上,sizeof(AlignedUnion) 会是 8。
// 因为最大的成员是double (8字节),它的对齐要求是8字节。
// 所以整个union的大小必须是8的倍数,且至少能容纳8字节。

union AnotherUnion {
    char c;   // 1 byte, 默认对齐1字节
    long long ll; // 8 bytes, 默认对齐8字节
    short s;  // 2 bytes, 默认对齐2字节
};
// sizeof(AnotherUnion) 也会是 8。
// 最大的成员是long long (8字节),对齐要求是8。
// 即使char和short都在,union大小也必须是8的倍数。

这种对齐规则有时会让人觉得有点“浪费”,比如一个

char
long long
的联合体,实际上只有
long long
在使用时才真正需要8字节,但整个联合体却固定占8字节。但这正是为了CPU访问效率和跨平台兼容性所做的权衡。编译器在幕后为我们处理了这些细节,确保了程序的正确性和效率。

联合体在C++实际编程中有哪些典型应用场景?

联合体虽然在现代C++中被一些更安全的替代方案(如

std::variant
)所取代,但在某些特定场景下,它依然是简洁高效的选择,尤其是在需要底层控制或与C语言兼容时。

  • 节省内存: 这是联合体最直接、最核心的应用。当你知道在程序的不同阶段,同一块内存区域需要存储不同类型但不会同时使用的信息时,联合体是理想选择。例如,在一个消息处理系统中,一个消息结构可能包含不同类型的负载(文本、图片数据、控制指令),但每次只发送一种。使用联合体可以避免为每种可能的负载类型都分配独立的内存,从而显著减少内存占用

  • 变体类型 (Variant Types): 联合体常与枚举(enum)结合使用,创建“变体”类型,即一个数据结构可以表示多种不同的数据类型。枚举用于指示当前联合体中哪个成员是活跃的,从而安全地访问正确的成员。

    enum DataType {
        INT_TYPE,
        DOUBLE_TYPE,
        STRING_TYPE // 假设是固定大小的字符串
    };
    
    struct Data {
        DataType type; // 用于指示当前活跃的类型
        union {
            int iVal;
            double dVal;
            char sVal[20]; // 固定大小字符串,避免动态内存分配的复杂性
        } value;
    };
    
    // 使用示例:
    Data myData;
    myData.type = INT_TYPE;
    myData.value.iVal = 123;
    
    // 或者
    myData.type = DOUBLE_TYPE;
    myData.value.dVal = 3.14;
    
    // 访问时需要检查类型:
    if (myData.type == INT_TYPE) {
        // 使用 myData.value.iVal
    }

    这种模式在实现类似

    std::variant
    (C++17之前)或在C语言中处理异构数据时非常常见,提供了一种类型安全的变通方案。

  • 底层数据操作/类型转换(通常不推荐,但仍存在于遗留代码): 联合体有时被用来“欺骗”编译器,将一块内存区域按不同的类型解释,从而实现某些底层操作。例如,将

    float
    的位模式作为
    unsigned int
    来访问,以检查其内部表示。

    union FloatToInt {
        float f;
        unsigned int i;
    };
    
    FloatToInt converter;
    converter.f = 3.14f;
    // 现在 converter.i 包含了 3.14f 的原始位模式
    // 注意:这种做法在C++中严格来说是类型双关 (type punning),
    // 除了C++20引入的union成员访问规则外,通常会导致未定义行为 (UB)。
    // 推荐的现代C++做法是使用 memcpy 或 C++20的 std::bit_cast 来进行这种位模式转换,
    // 以确保行为是明确和可移植的。

    我个人在维护一些老代码时,确实见过这种用法,虽然现在看来有些“野蛮”,但在那个年代,它提供了一种绕过类型系统的直接访问方式。不过,现代C++已经提供了更安全、更明确的机制来处理这些需求,因此在新的代码中应尽量避免这种直接的联合体类型双关。

相关专题

更多
C语言变量命名
C语言变量命名

c语言变量名规则是:1、变量名以英文字母开头;2、变量名中的字母是区分大小写的;3、变量名不能是关键字;4、变量名中不能包含空格、标点符号和类型说明符。php中文网还提供c语言变量的相关下载、相关课程等内容,供大家免费下载使用。

387

2023.06.20

c语言入门自学零基础
c语言入门自学零基础

C语言是当代人学习及生活中的必备基础知识,应用十分广泛,本专题为大家c语言入门自学零基础的相关文章,以及相关课程,感兴趣的朋友千万不要错过了。

612

2023.07.25

c语言运算符的优先级顺序
c语言运算符的优先级顺序

c语言运算符的优先级顺序是括号运算符 > 一元运算符 > 算术运算符 > 移位运算符 > 关系运算符 > 位运算符 > 逻辑运算符 > 赋值运算符 > 逗号运算符。本专题为大家提供c语言运算符相关的各种文章、以及下载和课程。

352

2023.08.02

c语言数据结构
c语言数据结构

数据结构是指将数据按照一定的方式组织和存储的方法。它是计算机科学中的重要概念,用来描述和解决实际问题中的数据组织和处理问题。数据结构可以分为线性结构和非线性结构。线性结构包括数组、链表、堆栈和队列等,而非线性结构包括树和图等。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

256

2023.08.09

c语言random函数用法
c语言random函数用法

c语言random函数用法:1、random.random,随机生成(0,1)之间的浮点数;2、random.randint,随机生成在范围之内的整数,两个参数分别表示上限和下限;3、random.randrange,在指定范围内,按指定基数递增的集合中获得一个随机数;4、random.choice,从序列中随机抽选一个数;5、random.shuffle,随机排序。

597

2023.09.05

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

523

2023.09.20

c语言get函数的用法
c语言get函数的用法

get函数是一个用于从输入流中获取字符的函数。可以从键盘、文件或其他输入设备中读取字符,并将其存储在指定的变量中。本文介绍了get函数的用法以及一些相关的注意事项。希望这篇文章能够帮助你更好地理解和使用get函数 。

639

2023.09.20

c数组初始化的方法
c数组初始化的方法

c语言数组初始化的方法有直接赋值法、不完全初始化法、省略数组长度法和二维数组初始化法。详细介绍:1、直接赋值法,这种方法可以直接将数组的值进行初始化;2、不完全初始化法,。这种方法可以在一定程度上节省内存空间;3、省略数组长度法,这种方法可以让编译器自动计算数组的长度;4、二维数组初始化法等等。

599

2023.09.22

C++ 单元测试与代码质量保障
C++ 单元测试与代码质量保障

本专题系统讲解 C++ 在单元测试与代码质量保障方面的实战方法,包括测试驱动开发理念、Google Test/Google Mock 的使用、测试用例设计、边界条件验证、持续集成中的自动化测试流程,以及常见代码质量问题的发现与修复。通过工程化示例,帮助开发者建立 可测试、可维护、高质量的 C++ 项目体系。

3

2026.01.16

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 6.8万人学习

C 教程
C 教程

共75课时 | 4万人学习

C++教程
C++教程

共115课时 | 12.4万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号