1. 程式人生 > >union的用法以及struct的初始化

union的用法以及struct的初始化

在C/C++程式的編寫中,當多個基本資料型別或複合資料結構要佔用同一片記憶體時,我們要使用聯合體;當多種型別,多個物件,多個事物只取其一時(我們姑且通俗地稱其為“n 選1”),我們也
可以使用聯合體來發揮其長處。首先看一段程式碼: union myun 
{
struct { int x; int y; int z; }u; 
int k; 
}a; 
int main() 

a.u.x =4;
a.u.y =5; 
a.u.z =6; 
a.k = 0; 
printf("%d %d %d\n",a.u.x,a.u.y,a.u.z);
return 0;
}
union型別是共享記憶體的,以size最大的結構作為自己的大小,這樣的話,myun這個結構就包含u這個結構體,而大小也等於u這個結構體 的大小,在記憶體中的排列為宣告的順序x,y,z從低到高,然後賦值的時候,在記憶體中,就是x的位置放置4,y的位置放置5,z的位置放置6,現在對k賦 值,對k的賦值因為是union,要共享記憶體,所以從union的首地址開始放置,首地址開始的位置其實是x的位置,這樣原來記憶體中x的位置就被k所賦的 值代替了,就變為0了,這個時候要進行列印,就直接看記憶體裡就行了,x的位置也就是k的位置是0,而y,z的位置的值沒有改變,所以應該是0,5,6 再看兩個試題: 試題一:編寫一段程式判斷系統中的CPU 是Little endian 還是Big endian 模式?
分析:
作 為一個計算機相關專業的人,我們應該在計算機組成中都學習過什麼叫Little endian 和Big endian。Little endian 和Big endian 是CPU 存放資料的兩種不同順序。對於整型、長整型等資料型別,Big endian 認為第一個位元組是最高位位元組(按照從低地址到高地址的順序存放資料的高位位元組到低位位元組);而Little endian 則相反,它認為第一個位元組是最低位位元組(按照從低地址到高地址的順序存放資料的低位位元組到高位位元組)。
例如,假設從記憶體地址0x0000 開始有以下資料:
0x12 0x34 0xab 0xcd
如 果我們去讀取一個地址為0x0000 的四個位元組變數,若位元組序為big-endian,則讀出結果為0x1234abcd;若位元組序位little-endian,則讀出結果為 0xcdab3412。如果我們將0x1234abcd 寫入到以0x0000 開始的記憶體中,則Little endian 和Big endian 模式的存放結果如下:

地址               0x0000 0x0001 0x0002 0x0003
big-endian         0x12   0x34   0xab   0xcd
little-endian      0xcd   0xab 0x34   0x12
一般來說,x86 系列CPU 都是little-endian 的位元組序,PowerPC 通常是Big endian,還有的CPU 能通過跳線來設定CPU 工作於Little endian 還是Big endian 模式。
解答:
顯然,解答這個問題的方法只能是將一個位元組(CHAR/BYTE 型別)的資料和一個整型資料存放於同樣的記憶體
開始地址,通過讀取整型資料,分析CHAR/BYTE 資料在整型資料的高位還是低位來判斷CPU 工作於Little

endian 還是Big endian 模式。得出如下的答案:
typedef unsigned char BYTE;
int main(int argc, char* argv[])
{
unsigned int num,*p;
p = #
num = 0;
*(BYTE *)p = 0xff;
if(num == 0xff)
{
printf("The endian of cpu is little\n");
}
else //num == 0xff000000
{
printf("The endian of cpu is big\n");
}
return 0;
}
除了上述方法(通過指標型別強制轉換並對整型資料首位元組賦值,判斷該賦值賦給了高位還是低位)外,還有沒
有更好的辦法呢?我們知道,union 的成員本身就被存放在相同的記憶體空間(共享記憶體,正是union 發揮作用、做貢獻的去處),因此,我們可以將一個CHAR/BYTE 資料和一個整型資料同時作為一個union 的成員,得出
如下答案:
int checkCPU()
{
{
union w
{
int a;
char b;
} c;
c.a = 1;
return (c.b == 1);
}
}

實現同樣的功能,我們來看看Linux 作業系統中相關的原始碼是怎麼做的:
static union { char c[4]; unsigned long mylong; } endian_test = {{ 'l', '?', '?', 'b' } };

#define ENDIANNESS ((char)endian_test.mylong)

Linux 的核心作者們僅僅用一個union 變數和一個簡單的巨集定義就實現了一大段程式碼同樣的功能!由以上一段程式碼我們可以深刻領會到Linux 原始碼的精妙之處!(如果ENDIANNESS=’l’表示系統為little endian,
為’b’表示big endian )
試題二:假設網路節點A 和網路節點B 中的通訊協議涉及四類報文,報文格式為“報文型別欄位+報文內容的結構體”,四個報文內容的結構體型別分別為STRUCTTYPE1~ STRUCTTYPE4,請編寫程式以最簡單的方式組
織一個統一的報文資料結構。

分析:
報文的格式為“報文型別+報文內容的結構體”,在真實的通訊中,每次只能發四類報文中的一種,我們可以將四類報文的結構體組織為一個union(共享一段記憶體,但每次有效的只是一種),然後和報文型別欄位統一組織成一個報文資料結構。
解答:
根據上述分析,我們很自然地得出如下答案:
typedef unsigned char BYTE;
//報文內容聯合體
typedef union tagPacketContent
{
STRUCTTYPE1 pkt1;
STRUCTTYPE2 pkt2;
STRUCTTYPE3 pkt1;
STRUCTTYPE4 pkt2;
}PacketContent;
//統一的報文資料結構
typedef struct tagPacket
{
BYTE pktType;
PacketContent pktContent;

}Packet;

寫了一個小例子,將union和struct相結合。

#include<stdio.h>
struct end{
        int end;
        union{
                        char c;
                        int k;
                }myun;

     /*union myun{

char c;

int k;

}a;struct中的union可以用這兩種方法來宣告*/
};
int main()
{
        struct end end1={
                .end=24,
                .myun={'k'},
        };//注意struct的初始化,各個成員初始化完畢之後使用‘,’而不是‘;’。
        printf("%d %c %d\n",end1.end,end1.myun.c,end1.myun.k);
        return 0;

}

相關推薦

union用法以及struct初始

在C/C++程式的編寫中,當多個基本資料型別或複合資料結構要佔用同一片記憶體時,我們要使用聯合體;當多種型別,多個物件,多個事物只取其一時(我們姑且通俗地稱其為“n 選1”),我們也 可以使用聯合體來發揮其長處。首先看一段程式碼: union myun  { struc

Linux系統--CentOS7下Mysql(docker)映象建立,使用者,表以及資料初始

1.docker下載(環境為centos7)yum install docker-engine2.下載完成後啟動docker使用命令:service docker start3.為docker下載映象提速curl -sSL https://get.daocloud.io/da

Pytorch 引數初始以及Xavier初始

def _initialize_weights(self): # print(self.modules()) for m in self.modules(): print(m) if isinstance(m,

Spring+Spring MVC+Mybatis整合配置AOP不生效的解決方案以及Bean初始重複載入兩次(疑難雜症)

之前上班做spring+spring mvc +hibernate開發, 2年之久不做想複習一下aop的使用,結果配置遇見aop不生效,解決而記錄! 先上程式碼直接看反例效果會明顯: 首先看一下我的程式碼的包路徑: 接下來看Spring-MVC的配置檔案部分程式碼:

各種容器的初始以及相互初始

integer list bsp new res 容器 初始 初始化 nbsp 1D-->1D List: new ArrayList<>(1D_list) 1D List添加 2D List List<List<Integer&

vmware安裝centos6.5以及初始

centosVMware安裝centos6.5習慣: 新建虛擬機-->自定義-->稍後安裝系統-->linux(其他linux 2.6x內核64位) 然後配置,記得加上安裝鏡像,網絡配置位橋接,內存給20G左右就可以。 開機-->選第一條-->skip-->

idea中如何配置git以及在idea中初始git

tle bsp nbsp 技術 alt 以及 ide ash 位置 idea中如何配置git以及在idea中初始化git呢: 參考此博文: http://blog.csdn.net/qq_28867949/article/details/73012300 *為了這個問題

Jquery 頁面初始常用的三種方法以及Jquery 發送ajax 請求

tree error 渲染 erro 發生 har 初始化 nload 事情 第一種 $(document).ready(function(){ //文檔就緒事件 }); 第二種是第一種的簡略寫法,效果上和第一種是等效的。 $(function(){ //文檔加載事

MySQL初始以及客戶端工具的使用

sock 詳細信息 linux form 關系型 orm ctr create 主機名                       MySQL初始化以及客戶端工具的使用                                             作者:尹正傑 版

java變量和作用域以及成員變量的默認初始

作用 內存 bool 字段 效果 spa height 表示 創建 Java中的變量有成員變量和局部變量,定義在類中方法之外的變量成為成員變量或者成員字段(域),表示一個類所具有的屬性,定義為類的成員變量的變量的作用於是整個類,該變量在定義的時候不需要初始化,在使用前jav

python list的深拷貝與淺拷貝-以及初始空白list的方法(2)

src 分享圖片 [1] 深拷貝 pen net .com 空白 tails 其實python對於深拷貝與淺拷貝的處理方式依然是很靈活的,甚至有一個專門的工具包copy來做個事情 來看以下這組list的拷貝方法: 已知lista是一個已經定義好了的list listb=l

struct結構體指定初始項目

錯位 發生 頻繁 常見 語法 float 其中 標準 硬件 標準C的標記化結構初始化語法在標準C中(C89)結構標準初始化是用{}來實始化,在C99的版本,采用了采用可讀性更強的標記化實始化,這在LINUX內核和驅動很為常見。這是ISOC99的用法CPrimer P

Webkit初始以及載入URL過程中各種對象的建立時序以及DOM樹的建立詳情分析

沒有 chrom 本地 .net 時間 詳情 request avi ng-click ? ? ? ? 眾所周知,Webkit須要創建DOM樹。為此它須要創建Web

union, enum, and struct, 以及結構填充和位字段實現。

表示 bits 聯合 follow ++ alignment necessary only lds Table 4-9 Compiler storage of data objects by byte alignment TypeBytesAlignmen

mysql 安裝成功以及第一次安裝成功初始密碼操作

sql升級 mar 命令 my.ini windows mysql5.7 display 圖片 直接 一 把文件解壓到一個目錄下 這是解壓後的目錄 將my.ini文件考進去 雙擊打開my.ini 找到這兩行更改成自己的解壓路徑保存 右鍵此電腦屬性

***靜態成員的定義及初始 for c++ for新用法

用法 靜態數組 In const AI code 引用 clu RR 靜態成員的初始化要在類外不然報錯error: ISO C++ forbids in-class initialization of non-const static member ‘***‘ 但是聲明為

Java構造方法、成員變量初始以及靜態成員變量初始三者的先後順序是什麽樣的?

java學習 程序 tar 變量初始化 ref 優先 靜態 上海 優先級 【Java筆試真題】:構造方法、成員變量初始化以及靜態成員變量初始化三者的先後順序是什麽樣的? 【解答】:當類第一次被加載的時候,靜態變量會首先初始化,接著編譯器會把實例變量初始化為默認值,然後執行構

C++:建構函式的初始列表,以及初始的順序

建構函式的初始化值列表: 對於物件的資料成員而言,初始化和賦值是有區別的。 當資料成員是 const 、引用,或者屬於某種未提供預設建構函式的類型別的話,就必須通過建構函式的初始值列表為這些成員提供初始值,否則就會引發錯誤。 // 錯誤:ci 和 ri 必須初始化 CobstRef::C

【AI系列】<3>生成神經網路以及初始詳細過程

內容簡介 本文將會介紹神經網路的生成過程,通過生成一個[2, 5, 3, 1]的神經網路,詳細講解每一步驟的運算過程。[2, 5, 3, 1] 表示:兩個輸入,第一層 5各神經元,第二層3個神經元,第三層 1個神經元。 生成輸入資料 np.random.s

Java基礎——靜態程式碼塊、構造程式碼塊、建構函式以及Java類初始順序

閱讀目錄 建構函式 構造程式碼塊 靜態程式碼塊 Java類初始化順序 靜態程式碼塊:用staitc宣告,jvm載入類時執行,僅執行一次 構造程式碼塊:類中直接用{}定義,每一次建立物件時執行。 執行順序優先順序:靜態塊,main(),構造塊,構造方法。