曾经在学校学习C语言的时候一直搞不懂那个共用体union有什么用的。工作之后才发现它的一些妙用,现举比例如以下:

1. 为了方便看懂代码。

比方说想写一个3 * 3的矩阵,能够这样写:
[
注:以下用红色部分标记的地方是后来加入上去的,谢谢yrqing718的提醒!]

  1. struct
     Matrix
  2. {
  3. union
  4. {
  5. struct
  6. {
  7. float
     _f11, _f12, _f13, _f21, _f22, _f23, _f31, _f32, _f33;
  8. };
  9. float
     f[3][3];
  10. }_matrix;
  11. };
  12. struct
     Matrix m;

这两个东西共同使用同样的空间,所以没有空间浪费,在须要总体用矩阵的时候能够用
m._matrix.f (比方说传參,或者是总体赋值等);须要用当中的几个元素的时候能够用m._matrix._f11那样能够避免用m.f[0][0](这样不大直观,并且easy出错)。

2. 用在强制类型转换上(比强制类型转换更加easy看懂)
以下举几个样例:

(1). 推断系统用的是big endian 还是 little endian(其定义大家能够到网上查相关资料,此略)

  1. #define TRUE 1
  2. #define FALSE 0
  3. #define BOOL int
  4. BOOL
     isBigEndian()
  5. {
  6. int
     i = 1;   /* i = 0x00000001*/
  7. char
     c = *(char
     *)&i; /* 注意不能写成 char c = (char)i; */
  8. return
     (int
    )c != i;
  9. }

假设是little endian字节序的话,那个i = 1;的内存从小到大依次放的是:0x01 0x00 0x00 0x00,如是,依照i的起始地址变成依照char *方式(1字节)存取,即得c = 0x01;
反之亦然

或许看起来不是非常清晰,以下来看一下这个:

  1. BOOL
     isBigEndian()
  2. {
  3. union
  4. {
  5. int
     i;
  6. char
     c;
  7. }test;
  8. test.c = 2;
  9. return
     test.i != 2;
  10. }

这里用的是union来控制这个共享布局,有个知识点就是union里面的成员c和i都是从低地址開始对齐的。相同能够得到如此结果,并且不用转换,清晰一些。

什么,不认为清晰??那再看以下的样例:

(2). 将little endian下的long long类型的值换成 big endian类型的值。已经知道系统提供了以下的api:long htonl(long lg);作用是把全部的字节序换成大端字节序。因此得出以下做法:

  1. long
     long
     htonLL(long
     long
     lg)
  2. {
  3. union
  4. {
  5. struct
  6. {
  7. long
     low;
  8. long
     high;
  9. }val_1;
  10. long
     long
     val_2;
  11. }val_arg, val_ret;
  12. if
    ( isBigEndian() )
  13. return
     lg;
  14. val_arg.val_2 = lg;
  15. val_ret.val_1.low = htonl( val_arg.val_1.high );
  16. val_ret.val_1.high = htonl( val_arg.val_1.low );
  17. return
     val_ret.val_2;
  18. }

仅仅要把内存结构的草图画出来就比較easy明确了。

(3).为了理解c++类的布局,再看以下一个样例。有例如以下类:

  1. class
     Test
  2. {
  3. public
    :
  4. float
     getFVal(){ return
     f;}
  5. private
    :
  6. int
     i;
  7. char
     c;
  8. float
     f;
  9. };
  10. Test t;

不能在类Test中添加代码,给对象中的f赋值7.0f.

  1. class
     Test_Cpy
  2. {
  3. public
    :
  4. float
     getVal(){ return
     f;}
  5. float
     setVal(float
     f){ this
    ->f = f;}
  6. private
    :
  7. int
     i;
  8. char
     c;
  9. float
     f;
  10. };
  11. ....
  12. int
     main()
  13. {
  14. Test t;
  15. union
  16. {
  17. Test t1,
  18. Test_Cpy t2;
  19. }test;
  20. test.t2.setVal(7.0f);
  21. t = test.t1;
  22. assert( t.getVal() == 7.0f );
  23. return
     0;
  24. }

说明:由于在添加类的成员函数时候,那个类的对象的布局基本不变。因此能够写一个与Test类一样结构的类Test_Cpy,而多了一个成员函数setVal,再用uinon结构对齐,就能够给私有变量赋值了。(这样的方法在有虚机类和虚函数机制时可能失灵,故不可移植)至于具体的讨论,网上有,这个样例在实际中没实用途,仅仅是用来考察这个内存布局的使用而已.

union在操作系统底层的代码中用的比較多,由于它在内存共赏布局上方便且直观。所以网络编程,协议分析,内核代码上有一些用到union都比較好懂,简化了设计。

(C语言)共用体union的使用方法举例的更多相关文章

  1. C语言共用体union

    union共用体说明: 当一个共用体被声明时, 编译程序自动地产生一个变量, 其长度为联合中最大的变量长度的整数倍. 比如union中有{int x; double x1; char name[10] ...

  2. C语言共用体、大小端、枚举

    1.共用体和结构体的相同和不同 (1)相同点就是操作语法几乎相同.(2)不同点是本质上的不同.struct是多个独立元素(内存空间)打包在一起:union是一个元素(内存空间)的多种不同解析方式. # ...

  3. C语言 共用体

    //共用体 union #define _CRT_SECURE_NO_WARNINGS #include<stdio.h> #include<stdlib.h> #includ ...

  4. C语言共用体(Union)

    通过前面的讲解,我们知道结构体(Struct)是一种构造类型或复杂类型,它可以包含多个类型不同的成员.在C语言中,还有另外一种和结构体非常类似的语法,叫做共用体(Union),它的定义格式为: uni ...

  5. 共用体union

    union共用体名 { 类型名 成员名1: 类型名 成员名2: …… 类型名 成员名n; } 由于各个成员变量在内存中都使用同一段存储空间,因此共用体变量的长度等于最长的成员长度,共用体的访问方式和结 ...

  6. 共用体 union

    共用体是一种数据格式,能够存储不同的数据类型,但只能同时存储其中的一种类型. union one4all { int int_val; double double_val; long long_val ...

  7. C结构体struct 和 共用体union的使用测试

    #include <stdio.h> struct { char name[10]; char sex; char job; int num; union{ //联合只能共用同一个内存 i ...

  8. 以位为单位存储标志-共用体-union

    一.程序的结构如下: typedef union _KEYST     {         struct         {             uint8 Key1_Flag :1;//表示第0 ...

  9. C语言联合体(union)的使用方法及其本质-union

    转载自:https://blog.csdn.net/si_zhou_qun_84342712/article/details/53187106 1.联合体union的基本特性——和struct的同与不 ...

随机推荐

  1. NET MVC过滤器

    NET MVC过滤器 本系列目录:ASP.NET MVC4入门到精通系列目录汇总 在ASP.NET MVC中有四种过滤器类型

  2. Java Math的floor,round,ceil函数小结

    转自 http://blog.csdn.net/foart/article/details/4295645 floor 返回不大于的最大整数 round 则是4舍5入的计算,入的时候是到大于它的整数( ...

  3. SQL 2008执行语句遇到内存不足(1)——error 701

    原文:SQL 2008执行语句遇到内存不足(1)--error 701 转自:http://blogs.msdn.com/b/apgcdsd/archive/2011/01/17/sql-2008-e ...

  4. hibernate 多对多 最佳实践

    首先 看看我们 ER 画画 :盖  一对一 .一对多 .多对多 的关系 watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvem91dG9uZ3l1YW4=/fo ...

  5. zigbee、profile、cluster、 endpoint、

    1.引用ZigBee联盟的说法 Profile: a collection of device descriptions, which together form a cooperative appl ...

  6. uva 10635 Prince and Princess(LCS成问题LIS问题O(nlogn))

    标题效果:有两个长度p+1和q+1该序列.的各种元素的每个序列不是相互同.并1~n^2之间的整数.个序列的第一个元素均为1. 求出A和B的最长公共子序列长度. 分析:本题是LCS问题,可是p*q< ...

  7. Hadoop 它们的定义Writable NullpointerException

    Hadoop周边环境:Hadoop2.4 定义中的Hadoop的Writable时间,有时你需要使用数组,而不是简单的单一值或串.例如,下面的代码: package test; import java ...

  8. 如何定义AIDL跨进程间通信

    当进程A要去调用进程B中的service时,并实现通信,我们通常都是通过AIDL来操作的 projectA: 首先在我们自己的包com.wzp.aidlservice中创建一个RemoteServic ...

  9. 设计模式(Facade)状态(注意事项)

    外观模式(Facade),子系统的一组接口提供一个一致的界面,该模式定义了一个高层次接口,这个接口使得这一子系统更加easy采用. 外观模式完美地体现了依赖反转原则,迪米特法则的想法,式之中的一个. ...

  10. ImageView建立selector在录音中遇到的小问题及解决方案

    随着两张照片做了一个selector,采用ImageView的src要么background采用selector当点击,总不会出现点击效果,这就是为什么?经过一番折腾,后来发现"揭秘&quo ...