当前位置: 首页 > news >正文

C语言基础系列【15】union 共用体

博主介绍:程序喵大人

  • 35- 资深C/C++/Rust/Android/iOS客户端开发
  • 10年大厂工作经验
  • 嵌入式/人工智能/自动驾驶/音视频/游戏开发入门级选手
  • 《C++20高级编程》《C++23高级编程》等多本书籍著译者
  • 更多原创精品文章,首发gzh,见文末
  • 👇👇记得订阅专栏,以防走丢👇👇
    😃C++基础系列专栏
    😃C语言基础系列

union是一种特殊的数据结构,union在相同的内存位置存储不同类型的数据。union中的所有成员共享同一块内存空间,因此在任何时候,union只能存储其成员中的一个值。

示例代码

#include <iostream>

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

int main() {
    Data data;

    // 存储整数
    data.i = 42;
    std::cout << "Integer: " << data.i << std::endl;

    // 存储浮点数(注意:会覆盖之前的整数数据)
    data.f = 3.14f;
    std::cout << "Float: " << data.f << std::endl;

    // 存储字符串(注意:会覆盖之前的浮点数数据)
    strcpy(data.str, "Hello, Union!");
    std::cout << "String: " << data.str << std::endl;

    return 0;
}

在这个示例中,我们定义了一个Data union,它包含三个成员:

  • i(整型)
  • f(浮点型)
  • str(字符数组)

然后演示了如何向union中存储不同类型的数据。

请注意,由于union的所有成员共享内存,因此同时访问多个成员可能会导致未定义行为。下面详细介绍。

共用体的定义与基本用法

共用体的定义方式与结构体类似,但使用union关键字。

共用体可以包含多个成员,但任何时候只能有一个成员带有有效值。

因为共用体的所有成员共享同一块内存区域,当给某个成员赋值时,会覆盖该内存区域中之前存储的数据。

定义共用体的语法如下

union 共用体名 {
成员类型1 成员名1;
成员类型2 成员名2;
...
    成员类型n 成员名n;
};

其中,共用体名是可选的,用于标识该共用体类型。成员类型成员名则定义了共用体中包含的成员及其类型。

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

这个例子,定义了一个名为Data的共用体,它包含三个成员:一个整型变量i,一个浮点型变量f,以及一个字符数组str

使用共用体

#include <stdio.h>
#include <string.h>

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

int main() {
    union Data data;

    // 给整型成员赋值
    data.i = 10;
    printf("data.i: %d\n", data.i);

    // 给浮点型成员赋值(会覆盖之前的整数值)
    data.f = 220.5;
    printf("data.f: %.5f\n", data.f);

    // 给字符数组成员赋值(会覆盖之前的浮点数值)
    strcpy(data.str, "Hello, Union!");
    printf("data.str: %s\n", data.str);

    return 0;
}

在这个示例中,我们定义了一个Data类型的共用体变量data,并分别给它的整型、浮点型和字符数组成员赋值。由于共用体的成员共享同一块内存,因此每次赋值都会覆盖之前存储的数据。

共用体的内存占用

共用体占用的内存大小等于其最大成员所占用的内存大小。

因为共用体的所有成员都存储在同一个内存位置,所以只需要足够的空间来存储最大的成员即可。

示例

#include <stdio.h>

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

int main() {
    printf("Size of union Data: %zu bytes\n", sizeof(union Data));
    return 0;
}

在这个示例中,我们打印了Data类型共用体的内存大小。由于字符数组str是成员中占用空间最大的(20个字节),因此Data类型共用体的内存大小也是20个字节。

共用体的应用场景

共用体在C语言中有多种应用场景:

  1. 节省内存
    当需要在同一个内存位置存储多种类型的数据时,可以使用共用体来节省内存。
    例如,在处理网络通信数据时,可能会根据协议的不同而需要解析不同类型的数据(如整型、浮点型、字符串等),此时可以使用共用体来存储这些数据。

  2. 类型转换
    共用体可以用于在不同类型之间进行转换。例如,在某些情况下,可能需要将一个整型数转换为浮点型数,或者将一个字符数组转换为整型数进行运算。虽然C语言提供了类型转换运算符,但在某些情况下使用共用体可能方便。

共用体与结构体的区别

  • 内存占用:结构体中的每个成员都占用独立的内存空间,而共用体中的所有成员共享同一块内存空间。
  • 数据访问:在结构体中,可以同时访问多个成员的值;而在共用体中,每次只能访问一个有效成员的值(即最后一次被赋值的成员)。
  • 用途:结构体通常用于表示具有固定结构和属性的数据记录(如学生信息、图书信息等),而共用体则更多地用于在不同类型的数据之间进行转换或节省内存。

共用体与结构体的结合使用

下面展示了如何将共用体与结构体结合使用来存储多种类型的数据和额外的属性信息:

#include <stdio.h>
#include <string.h>

// 定义一个共用体来存储不同类型的数据
union DataValue {
int intValue;
float floatValue;
char strValue[20];
};

// 定义一个结构体来存储数据值和额外的属性信息
struct DataRecord {
union DataValue value;
char type; // 用于标识存储的数据类型(如'i'表示整型,'f'表示浮点型,'s'表示字符串)
char description[50]; // 用于存储数据的描述信息
};

int main() {
    struct DataRecord record;

    // 存储整型数据
    record.value.intValue = 123;
    record.type = 'i';
    strcpy(record.description, "This is an integer value.");
    printf("Record 1: Type = %c, Value = %d, Description = %s\n", record.type, record.value.intValue, record.description);

    // 存储浮点型数据(会覆盖之前的整数值)
    record.value.floatValue = 456.78;
    record.type = 'f';
    strcpy(record.description, "This is a float value.");
    printf("Record 2: Type = %c, Value = %.2f, Description = %s\n", record.type, record.value.floatValue, record.description);

    // 存储字符串数据(会覆盖之前的浮点数值)
    strcpy(record.value.strValue, "Hello, Union!");
    record.type = 's';
    strcpy(record.description, "This is a string value.");
    printf("Record 3: Type = %c, Value = %s, Description = %s\n", record.type, record.value.strValue, record.description);

    return 0;
}

在这个示例中,我们定义了一个DataValue类型的共用体来存储不同类型的数据(整型、浮点型、字符串),又定义了一个DataRecord类型的结构体来存储数据值和额外的属性信息(如数据类型和描述信息)。

练习

  • 定义一个union,包含intfloatchar类型的成员。编写一个程序,允许用户选择数据类型(通过输入),然后根据选择输入相应的值,并输出该值。

码字不易,欢迎大家点赞关注评论,谢谢!


C++训练营

专为校招、社招3年工作经验的同学打造的1V1 C++训练营,量身定制学习计划、每日代码review,简历优化,面试辅导,已帮助多名学员获得offer!训练营介绍

相关文章:

  • Apache Doris 实现毫秒级查询响应
  • 【RabbitMQ业务幂等设计】RabbitMQ消息是幂等的吗?
  • Ubuntu安装PostgreSQL
  • 城市地质安全专题连载⑦ | 加强国土空间规划管控,规避城市地质安全风险
  • 跟着李沐老师学习深度学习(十二)
  • javaSE学习笔记21-线程(thread)-锁(synchronized 与Lock)
  • 从零开始用STM32驱动DRV8301:无人机/机器人电机控制指南
  • 基于图扑 HT 可视化实现智慧地下采矿可视化
  • CentOS更换yum源
  • 安装MySQL9.1.0-winx64.msi的报错解决办法:Database initialization failed。(也适用9.2.0)
  • 基于spring的策略模式
  • 【树莓派Pico设备驱动】-MAX7219驱动8位7段数码管(基于SPI)
  • 微信小程序地图map全方位解析
  • Vue实战【后端返回ArrayBuffer时,前端如何处理并成功下载ArrayBuffer文件】
  • Hive JOIN过滤条件位置玄学:ON vs WHERE的量子纠缠
  • c#编程:LINQ是什么?
  • 关于docker及容器的了解学习记录
  • 【 Avalonia UI 语言国际化 I18n】图文结合教学,保姆级教学,语言国际化就是这么简单(.Net C#)
  • 谷粒商城学习笔记-13-配置git-ssh-配置代码免密提交
  • 自然语言处理:第九十二章 chatBI 经验(转载)
  • 一品威客旅游网站开发/免费推广引流平台推荐
  • 深圳疫情现在好像越来越严重/google搜索优化方法
  • wordpress模板排行榜/性能优化工具
  • 使用QQ做网站客服/国外搜索引擎排名百鸣
  • 做网站之前需要准备什么条件/苏州网站制作
  • 网站推广要我营业执照复印件/seo推广怎么做