久久国产成人av_抖音国产毛片_a片网站免费观看_A片无码播放手机在线观看,色五月在线观看,亚洲精品m在线观看,女人自慰的免费网址,悠悠在线观看精品视频,一级日本片免费的,亚洲精品久,国产精品成人久久久久久久

分享

從兩道經(jīng)典試題談C/C 中聯(lián)合體(union)的使用 - Inside SW,FW and HW R

 todaytomo 2007-05-30
從兩道經(jīng)典試題談C/C++中聯(lián)合體(union)的使用
宋寶華 21cnbao sweek@21cn.com
試題一:編寫一段程序判斷系統(tǒng)中的CPU是Little endian還是Big endian模式,?
分析:
作 為一個計算機相關(guān)專業(yè)的人,,我們應(yīng)該在計算機組成中都學(xué)習(xí)過什么叫Little endian和Big endian。Little endian和Big endian是CPU存放數(shù)據(jù)的兩種不同順序,。對于整型,、長整型等數(shù)據(jù)類型,Big endian認為第一個字節(jié)是最高位字節(jié)(按照從低地址到高地址的順序存放數(shù)據(jù)的高位字節(jié)到低位字節(jié)),;而Little endian則相反,,它認為第一個字節(jié)是最低位字節(jié)(按照從低地址到高地址的順序存放數(shù)據(jù)的低位字節(jié)到高位字節(jié))。
例如,,假設(shè)從內(nèi)存地址0x0000開始有以下數(shù)據(jù):
0x0000
0x0001
0x0002
0x0003
0x12
0x34
0xab
0xcd
如 果我們?nèi)プx取一個地址為0x0000的四個字節(jié)變量,,若字節(jié)序為big-endian,則讀出結(jié)果為0x1234abcd;若字節(jié)序位little- endian,,則讀出結(jié)果為0xcdab3412,。如果我們將0x1234abcd寫入到以0x0000開始的內(nèi)存中,則Little endian和Big endian模式的存放結(jié)果如下:
地址
0x0000
0x0001
0x0002
0x0003
big-endian
0x12
0x34
0xab
0xcd
little-endian
0xcd
0xab
0x34
0x12
一般來說,,x86系列CPU都是little-endian的字節(jié)序,,PowerPC通常是Big endian,還有的CPU能通過跳線來設(shè)置CPU工作于Little endian還是Big endian模式,。
解答:
顯 然,,解答這個問題的方法只能是將一個字節(jié)(CHAR/BYTE類型)的數(shù)據(jù)和一個整型數(shù)據(jù)存放于同樣的內(nèi)存開始地址,通過讀取整型數(shù)據(jù),,分析 CHAR/BYTE數(shù)據(jù)在整型數(shù)據(jù)的高位還是低位來判斷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;
}
除 了上述方法(通過指針類型強制轉(zhuǎn)換并對整型數(shù)據(jù)首字節(jié)賦值,判斷該賦值賦給了高位還是低位)外,,還有沒有更好的辦法呢,?我們知道,union的成員本身就 被存放在相同的內(nèi)存空間(共享內(nèi)存,,正是union發(fā)揮作用,、做貢獻的去處),因此,,我們可以將一個CHAR/BYTE數(shù)據(jù)和一個整型數(shù)據(jù)同時作為一個 union的成員,,得出如下答案:
int checkCPU()
{
 {
  union w
  {
   int a;
   char b;
  } c;
  c.a = 1;
  return (c.b == 1);
 }
}
實現(xiàn)同樣的功能,我們來看看Linux操作系統(tǒng)中相關(guān)的源代碼是怎么做的:
static union { char c[4]; unsigned long l; } endian_test = { { ‘l‘, ‘?‘, ‘?‘, ‘b‘ } };
#define ENDIANNESS ((char)endian_test.l)
Linux的內(nèi)核作者們僅僅用一個union變量和一個簡單的宏定義就實現(xiàn)了一大段代碼同樣的功能,!由以上一段代碼我們可以深刻領(lǐng)會到Linux源代碼的精妙之處,!
試題二:假設(shè)網(wǎng)絡(luò)節(jié)點A和網(wǎng)絡(luò)節(jié)點B中的通信協(xié)議涉及四類報文,報文格式為“報文類型字段+報文內(nèi)容的結(jié)構(gòu)體”,,四個報文內(nèi)容的結(jié)構(gòu)體類型分別為STRUCTTYPE1~ STRUCTTYPE4,,請編寫程序以最簡單的方式組織一個統(tǒng)一的報文數(shù)據(jù)結(jié)構(gòu)。
分析:
報文的格式為“報文類型+報文內(nèi)容的結(jié)構(gòu)體”,,在真實的通信中,,每次只能發(fā)四類報文中的一種,我們可以將四類報文的結(jié)構(gòu)體組織為一個union(共享一段內(nèi)存,,但每次有效的只是一種),,然后和報文類型字段統(tǒng)一組織成一個報文數(shù)據(jù)結(jié)構(gòu)。
解答:
根據(jù)上述分析,,我們很自然地得出如下答案:
typedef unsigned char BYTE;
 
//報文內(nèi)容聯(lián)合體
typedef union tagPacketContent
{
       STRUCTTYPE1 pkt1;
       STRUCTTYPE2 pkt2;
       STRUCTTYPE3 pkt1;
       STRUCTTYPE4 pkt2;
}PacketContent;
 
//統(tǒng)一的報文數(shù)據(jù)結(jié)構(gòu)
typedef struct tagPacket
{
       BYTE pktType;
    PacketContent pktContent;
}Packet;
總結(jié)
在C/C++程序的編寫中,,當多個基本數(shù)據(jù)類型或復(fù)合數(shù)據(jù)結(jié)構(gòu)要占用同一片內(nèi)存時,我們要使用聯(lián)合體(試題一是這樣的例證),;當多種類型,,多個對象,,多個事物只取其一時(我們姑且通俗地稱其為“n選1”),我們也可以使用聯(lián)合體來發(fā)揮其長處(試題二是這樣的例證),。


Trackback: http://tb./TrackBack.aspx?PostId=1054811

    本站是提供個人知識管理的網(wǎng)絡(luò)存儲空間,,所有內(nèi)容均由用戶發(fā)布,不代表本站觀點,。請注意甄別內(nèi)容中的聯(lián)系方式,、誘導(dǎo)購買等信息,謹防詐騙,。如發(fā)現(xiàn)有害或侵權(quán)內(nèi)容,,請點擊一鍵舉報。
    轉(zhuǎn)藏 分享 獻花(0

    0條評論

    發(fā)表

    請遵守用戶 評論公約

    類似文章 更多