Union内存分布

最近研究union,发现union内存分布挺有意思。

Union定义是什么?

Union是中文名是联合体,类似于struct,但是跟struct有很多区别,里面参数公用内存。

Union和struct的区别

①结构体(struct)中所有变量是"共存"的------优点是"有容乃大",全面;缺点是struct内存空间的分配是粗放的,不管用不用,全分配。

②而联合体(union)中是各变量是"互斥"的------缺点就是不够"包容";但优点是内存使用更为精细灵活,也节省了内存空间。

下面我们通过例子看看Union内存分布。

#include <stdio.h>
int main(){
    typedef union {
       long i;
       int k[4];
       char c;    
    }DATE;
    struct data
    {
        int cat;
        DATE cow;
        double dog;
    }too;
    DATE max;
    long a = 0;
    printf("%ld , %ld ,%ld\r\n",sizeof(struct data),sizeof(max),sizeof(struct data)+sizeof(max));
    printf("%p,%p ,%p\r\n", &max.i,&max.k,&max.c);
    printf("%ld\r\n",sizeof(long));
    return 0;
    
}

上面代码运行结果是

32 , 16 ,48
0x7ffc73484f50,0x7ffc73484f50 ,0x7ffc73484f50
8

看到第二行,访问Union变量里面的参数,地址都是一样的,完全就是共用一个内存首地址,并且各种变量名都可以同时使用,操作也是共同生效

内存分配

根据上面代码,union分配内存大小是16,因为long i 内存分配大小是8,int k[4]大小是16,char c分配大小是1,说明union分配大小是根据最大参数内存大小分配。

但是如果把int k[4]改成k[5]呢,上面结果会出现什么?

#include <stdio.h>
int main(){
    typedef union {
       long i;
       int k[5];
       char c;    
    }DATE;
    struct data
    {
        int cat;
        DATE cow;
        double dog;
    }too;
    DATE max;
    long a = 0;
    printf("%ld , %ld ,%ld\r\n",sizeof(struct data),sizeof(max),sizeof(struct data)+sizeof(max));
    printf("%p,%p ,%p\r\n", &max.i,&max.k,&max.c);
    printf("%ld\r\n",sizeof(long));
    return 0;
    
}

运行结果是

40 , 24 ,64
0x7fffbde34f10,0x7fffbde34f10 ,0x7fffbde34f10
8

此时union分配内存大小是24,前面不是说好了union分配大小是根据最大参数内存大小分配吗?

下面还有个概念,就是内存对齐,因为在64位机器上,long站8个字节,int k[5]占用20个字节,

但是由于内存对齐的原因,系统会将联合体的大小调整为最大成员大小的整数倍,即24个字节(20不是8的整数倍,所以会向上取整到24个字节)。

相关推荐
Dontla1 小时前
Rust泛型系统类型推导原理(Rust类型推导、泛型类型推导、泛型推导)为什么在某些情况必须手动添加泛型特征约束?(泛型trait约束)
开发语言·算法·rust
Ttang231 小时前
Leetcode:118. 杨辉三角——Java数学法求解
算法·leetcode
喜欢打篮球的普通人1 小时前
rust模式和匹配
java·算法·rust
java小吕布2 小时前
Java中的排序算法:探索与比较
java·后端·算法·排序算法
杜若南星2 小时前
保研考研机试攻略(满分篇):第二章——满分之路上(1)
数据结构·c++·经验分享·笔记·考研·算法·贪心算法
路遇晚风2 小时前
力扣=Mysql-3322- 英超积分榜排名 III(中等)
mysql·算法·leetcode·职场和发展
Neophyte06082 小时前
C++算法练习-day40——617.合并二叉树
开发语言·c++·算法
木向2 小时前
leetcode104:二叉树的最大深度
算法·leetcode
一个不喜欢and不会代码的码农2 小时前
力扣113:路径总和II
算法·leetcode