C · 进阶 | 联合体?看一遍就够了

简介: 结构体、枚举、联合体都是自定义类型

在这里插入图片描述
啊我摔倒了..有没有人扶我起来学习....


@TOC


前言

结构体、枚举、联合体都是自定义类型,结构体主要知识点结构体内存对齐可参考《C | 结构体内存对齐》枚举请参考《C | 枚举?看一遍就够了》


联合体

1. 联合体的定义

  • 与结构体非常类似
union Un
{
    char c;
    int i;
    double d;
};

2. 联合体的内存大小

  • 联合体所占内存又是多大呢?
int main()
{
    union Un un;
    printf("%d\n", sizeof(union Un));
    printf("%d\n", sizeof(un));

    return 0;
}
打印结果: 在这里插入图片描述可以看出来,是占用内存最大的成员 double决定的 8个字节。真的是这样吗?
  • 结构体又叫做共同体,顾名思义,它的成员都共用一个内存空间
int main()
{
    union Un un;
    printf("%p\n", &un);
    printf("%p\n", &(un.c));
    printf("%p\n", &(un.i));
    printf("%p\n", &(un.d));

    return 0;
}
打印结果: 在这里插入图片描述可以看到,确实是在同一个空间中。
  • 看起来联合体的内存就是成员中占用内存最大的那个决定,带着这个问题思考下述代码
在这里插入代码片

3. 联合体应用场景

  • 看完以上内容,铁汁们可能会觉得联合体根本没鸟用嘛!成员之间共同内存不就会互相覆盖了嘛!别急,咱们细细研究一下~
  • 我们设想一种情况,比如你是学生,有你的学号,储存在联合体中。将来有一天,你变成了老师,学号已经用不着了,取而代之的是职工号,这时联合体的作用不就出来啦~可以认为联合体的作用就是节省内存
  • 再来一个巧妙的用法!铁汁们还记得==判断大小端字节序==的方法吗,今天咱们利用联合体试试看
  1. 我们先用暴力的方法看看我们VS2019的编译器是什么字节序,先来段代码
int main()
{
    int num = 0x11223344;
    return 0;
}

然后开始调试,打开内存窗口看看。不会调试的铁汁可以看看《C | 实用调试技巧》
在这里插入图片描述可以看到是小端字节序存储方式

  1. 如果直接用代码能不能测试出来呢?先来段以前的方法
int main()
{
    int num = 1;
    char* p = (char*)#//int*

    if (*p == 1)
        printf("小端\n");
    else
        printf("大端\n");

    return 0;
}
打印结果: 在这里插入图片描述

在这里插入图片描述

  1. 接下来用联合体试试看
int check_sys()
{
    union Un
    {
        char c;
        int i;
    }u;
    u.i = 1;
    return u.c;
}

int main()
{
    int ret = check_sys();
    if (ret == 1)
        printf("小端\n");
    else
        printf("大端\n");
    return 0;
}
打印结果: 在这里插入图片描述

分析分析:
首先把1赋给ii是整型,占4个字节。然后用打印cc是字符型,占1个字节。而i又和c共用内存,小端存储的话1就位于低地址处,于是打印c就会是1
在这里插入图片描述

4. 联合体内存误区

  • 还是得谈谈联合体的内存,我们上述计算过一次联合体的内存
union Un
{
    char c;
    int i;
    double d;
};

在这里插入图片描述

  • 可是,真的取决于最大成员double的大小吗?我们来看看这段代码
union Un
{
    char arr[5];
    int i;
};
int main()
{
    printf("%d\n", sizeof(union Un));

    return 0;
}
打印结果: 在这里插入图片描述
==可以看出并不是 arr[5]所决定的 5个字节==
  • 所以,其实联合体也是需要对齐的,规则很简单,就是:

    1. 联合体的大小至少是最大成员的大小
    2. 当最大成员大小不是最大对齐数的整数倍时,就要对齐到最大对齐数的整数倍(注意,char arr[5]虽然是数组,但是它的对齐数取决于类型char,所以它的对齐数是1

总结

  • 联合体的用处和特点想必铁汁们都充分了解啦~如果铁汁对于内存对齐不够了解的话,还是需要学习一下《C | 结构体内存对齐》

在这里插入图片描述

相关文章
|
8月前
|
存储 编译器 C语言
【C语言必知必会 | 第三篇】顺序结构入门,这一篇就够了
【C语言必知必会 | 第三篇】顺序结构入门,这一篇就够了
495 0
|
存储 算法 C语言
20230806算法题(C语言)(适合专升本的同学和入门的小白)
20230806算法题(C语言)(适合专升本的同学和入门的小白)
|
存储 编译器 Linux
C++初阶之一篇文章让你掌握string类(了解和使用)(下)
3.5 string类对象的修改器(Modifiers) 函数名称 功能说明 operator+= 用于字符串的连接 append 在字符串末尾添加字符或字符序列
|
存储 C语言 C++
C语言拔高知识——指针的进阶(万字大文超详细)上
C语言拔高知识——指针的进阶(万字大文超详细)
163 0
C语言拔高知识——指针的进阶(万字大文超详细)上
|
存储 C语言
【简单地过一遍C语言基础部分】所有知识点,点到为止!(仅一万多字)(三)
就在前几天,C语言入门到进阶部分的专栏——《维生素C语言》终于完成了。全文共计十八个章节并附带三张笔试练习篇,美中不足的是,第一章和第二章是以截图形式展现的。由于本人一开始是在有道云笔记上写的初稿,当时想方便省事(有道云排版个人感觉确实比较美观)就直接以截图的形式完成了第一章和第二章。本人考虑到因为是截图,不能复制文中出现的代码,不方便读者进行复制粘贴,所以我打算重新写一下第一章和第一章的内容,并且重新进行了排版。
120 0
【简单地过一遍C语言基础部分】所有知识点,点到为止!(仅一万多字)(三)
|
存储 C语言
【简单地过一遍C语言基础部分】所有知识点,点到为止!(仅一万多字)(四)
就在前几天,C语言入门到进阶部分的专栏——《维生素C语言》终于完成了。全文共计十八个章节并附带三张笔试练习篇,美中不足的是,第一章和第二章是以截图形式展现的。由于本人一开始是在有道云笔记上写的初稿,当时想方便省事(有道云排版个人感觉确实比较美观)就直接以截图的形式完成了第一章和第二章。本人考虑到因为是截图,不能复制文中出现的代码,不方便读者进行复制粘贴,所以我打算重新写一下第一章和第一章的内容,并且重新进行了排版。
106 0
【简单地过一遍C语言基础部分】所有知识点,点到为止!(仅一万多字)(四)
|
存储 Java C#
【简单地过一遍C语言基础部分】所有知识点,点到为止!(仅一万多字)(二)
就在前几天,C语言入门到进阶部分的专栏——《维生素C语言》终于完成了。全文共计十八个章节并附带三张笔试练习篇,美中不足的是,第一章和第二章是以截图形式展现的。由于本人一开始是在有道云笔记上写的初稿,当时想方便省事(有道云排版个人感觉确实比较美观)就直接以截图的形式完成了第一章和第二章。本人考虑到因为是截图,不能复制文中出现的代码,不方便读者进行复制粘贴,所以我打算重新写一下第一章和第一章的内容,并且重新进行了排版。
156 0
【简单地过一遍C语言基础部分】所有知识点,点到为止!(仅一万多字)(二)
|
编译器 C语言
【简单地过一遍C语言基础部分】所有知识点,点到为止!(仅一万多字)(一)
就在前几天,C语言入门到进阶部分的专栏——《维生素C语言》终于完成了。全文共计十八个章节并附带三张笔试练习篇,美中不足的是,第一章和第二章是以截图形式展现的。由于本人一开始是在有道云笔记上写的初稿,当时想方便省事(有道云排版个人感觉确实比较美观)就直接以截图的形式完成了第一章和第二章。本人考虑到因为是截图,不能复制文中出现的代码,不方便读者进行复制粘贴,所以我打算重新写一下第一章和第一章的内容,并且重新进行了排版。
131 0
【简单地过一遍C语言基础部分】所有知识点,点到为止!(仅一万多字)(一)
|
存储 算法 编译器
C · 初阶 | 数组?看一遍就够了
数组(Array)是有序的元素序列。若将有限个类型相同的变量的集合命名,那么这个名称为数组名。组成数组的各个变量称为数组的分量,也称为数组的元素,有时也称为下标变量。用于区分数组的各个元素的数字编号称为下标。数组是在程序设计中,为了处理方便, 把具有相同类型的若干元素按有序的形式组织起来的一种形式。这些有序排列的同类数据元素的集合称为数组。
115 0
C · 初阶 | 数组?看一遍就够了
|
编译器
C · 进阶 | 枚举?看一遍就够了
结构体、枚举、联合体都是自定义类型
98 0
C · 进阶 | 枚举?看一遍就够了

热门文章

最新文章