C语言 共用体

  1. 概念
    • 在C语言中,共用体(Union)是一种特殊的数据类型。它可以在不同的时刻存储不同类型的数据,但所有成员共享同一块内存空间。这与结构体不同,结构体的每个成员都有自己独立的内存空间。
  2. 定义和声明
    • 定义
      • 共用体的定义形式与结构体相似,使用关键字union。例如:
        union Data {
            int num;
            char ch;
            float f;
        };
        
      • 这里定义了一个名为Data的共用体,它有三个成员:int类型的numchar类型的chfloat类型的f
    • 声明
      • 可以在定义共用体之后声明共用体变量。例如:
        union Data data;
        
      • 也可以在定义共用体的同时声明共用体变量:
        union Data {
            int num;
            char ch;
            float f;
        } data;
        
  3. 内存共享特性
    • 由于共用体的所有成员共享同一块内存空间,所以共用体的大小取决于其最大成员的大小。例如,在上述union Data中,如果int占4字节,char占1字节,float占4字节,那么union Data的大小为4字节。
    • 当给共用体的一个成员赋值时,会覆盖共用体中其他成员的值(因为它们共享内存)。例如:
      union Data data;
      data.num = 10;
      printf("num: %d\n", data.num);
      data.ch = 'a';
      printf("ch: %c\n", data.ch);
      // 此时,由于给ch赋值,num的值已经被覆盖,再次访问num的值是不确定的
      
  4. 使用场景
    • 节省内存空间
      • 在某些情况下,如果有一组数据,这些数据不会同时使用,就可以使用共用体来节省内存空间。例如,在一个设备驱动程序中,可能有一个数据结构用于存储设备的状态信息,这个状态信息可能是一个整数表示的错误代码,或者是一个字符表示的设备状态标识,使用共用体可以避免为这两种不同类型的状态信息分别分配独立的内存空间。
    • 数据类型转换
      • 共用体可以用于实现简单的数据类型转换。例如,将一个整数转换为字节数组。可以定义一个共用体,其中一个成员是整数,另一个成员是字符数组,通过给整数成员赋值,然后从字符数组成员中获取字节数据。
  5. 与结构体的区别
    • 内存分配
      • 结构体的每个成员都有自己独立的内存空间,结构体的大小是所有成员大小之和(考虑内存对齐)。而共用体所有成员共享同一块内存空间,其大小取决于最大成员的大小。
    • 数据存储和访问
      • 在结构体中,可以同时存储和访问多个成员的值。在共用体中,同一时刻只能存储和访问一个成员的值,因为成员共享内存,给一个成员赋值会影响其他成员的值。
  6. 注意事项
    • 数据覆盖风险
      • 由于共用体成员共享内存,在使用共用体时要特别注意数据覆盖的问题。如果不小心访问了已经被覆盖的成员的值,可能会得到错误的结果。
    • 类型兼容性
      • 在使用共用体进行数据转换或操作时,要确保数据类型之间的兼容性。例如,不能将一个不兼容的数据类型赋给共用体成员,否则可能会导致程序运行时错误。

共用体的大端和小端

  1. 大端和小端的概念

    • 大端(Big - Endian)
      • 在大端模式下,数据的高位字节存于低地址,低位字节存于高地址。例如,对于一个32位整数0x12345678,如果按照大端模式存储在内存中,内存地址从低到高依次存放的字节为0x120x340x560x78
    • 小端(Little - Endian)
      • 在小端模式下,数据的低位字节存于低地址,高位字节存于高地址。对于上述的0x12345678,按照小端模式存储时,内存地址从低到高依次存放的字节为0x780x560x340x12
  2. 共用体与大端小端的关系

    • 检测大端小端
      • 可以利用共用体的特性来检测系统是大端模式还是小端模式。例如:
        union Endian {
            int num;
            char c[4];
        };
        
        int main() {
            union Endian e;
            e.num = 0x12345678;
            if (e.c[0] == 0x78) {
                printf("小端模式\n");
            } else {
                printf("大端模式\n");
            }
            return 0;
        }
        
      • 在这个例子中,定义了一个共用体union Endian,它有一个int类型成员num和一个char类型数组c(长度为4,因为int通常为4字节)。由于共用体成员共享同一块内存空间,当给num赋值为0x12345678后,如果系统是小端模式,那么低地址(也就是c[0]所对应的地址)存放的是低位字节0x78;如果系统是大端模式,低地址存放的是高位字节0x12
    • 数据存储和访问的影响
      • 在共用体中,如果成员的数据类型跨越多个字节(如intfloat等),在不同的大端小端模式下,数据的存储和访问方式会有所不同。当从共用体的一个多字节成员转换到另一个成员(例如从int成员转换到char数组成员)时,需要考虑大端小端模式对数据字节顺序的影响。例如,在网络编程中,网络字节序通常采用大端模式,如果要将本地小端模式系统中的数据发送到网络上,就需要进行字节序转换,共用体可以在这种转换过程中起到一定的辅助作用。
    • 代码的可移植性
      • 在编写涉及共用体和多字节数据类型的代码时,要考虑大端小端模式的差异以确保代码的可移植性。如果不考虑这种差异,可能会导致数据在不同系统间传输或存储时出现错误。例如,在一个大端模式系统中编写的代码,如果直接在小端模式系统中运行,可能会对共用体中的数据做出错误的解释。
posted @ 2024-10-02 11:27  西北野狼  阅读(28)  评论(0编辑  收藏  举报