使用C語言位域的陷阱:大端與小端

今天在寫一個協(xié)議分析程序時著拭,使用了位域,因為協(xié)議的一個數(shù)據(jù)包有的參數(shù)并不是占據(jù)n個字節(jié)(bytes)跪但,而是占據(jù)n位(bits)峦萎。
比如有1個字節(jié)包含了4個參數(shù),分別占據(jù)的位數(shù)為3,1,1,2,1被环,我就在我的數(shù)據(jù)包結(jié)構(gòu)體中定義了如下位域

struct Packet
{
    // ...
    uint8_t a : 3;
    uint8_t b : 1;
    uint8_t c : 1;
    uint8_t d : 2;
    uint8_t e : 1;
    // ...
};

調(diào)試的時候發(fā)現(xiàn)結(jié)果不對详幽,然后寫了個測試

struct Widget
{
    uint8_t a : 4;
    uint8_t b : 3;
    uint8_t c : 1;

    void show()
    {
        printf("%d, %d, %d\n", a, b, c);
    }
};

void func()
{
    uint8_t x = 0b11100110;
    auto p = (Widget*)&x;
    p->show();  // output: 6, 6, 1
}

a表示前4位,b表示中間3位版姑,c表示后面1位迟郎,直觀地來看,a是1110(14)表制,b是011(3)控乾,c是0。但結(jié)果并非直觀看到的那樣壤短。
問題出在內(nèi)存布局方面慨仿,windows系統(tǒng)是小端布局,即低地址存放低字節(jié),也就是位域的順序是反過來的躲雅,即a是0110(6),b是110(6)相寇,c是1。
需要注意的是婆赠,大端小端是以字節(jié)為單位的,所以在內(nèi)存中休里,x并非存儲為
0 1 1 0 0 1 1 1 (從左向右地址依次增加)
而是針對位域而言赃承,低地址的成員(a)對應(yīng)的是字節(jié)的后半部分。
引用《C++ Primer》第5版19.8.1的說明

當(dāng)一個程序需要向其他程序或硬件設(shè)備傳遞二進制數(shù)據(jù)時拭嫁,通常會用到位域抓于。
位域在內(nèi)存中的布局是與機器相關(guān)的。

其中19.8這一小節(jié)的標(biāo)題是固有的不可移植的特性
因此使用位域的時候怕品,必須明確程序運行的機器是大端還是小端呕缭,再在代碼中定義各位域的具體順序。像Linux中/usr/include/netinet/tcp.h中定義TCP包結(jié)構(gòu)時就使用了位域

# if __BYTE_ORDER == __LITTLE_ENDIAN
    u_int16_t res1:4;
    u_int16_t doff:4;
    u_int16_t fin:1;
    u_int16_t syn:1;
    u_int16_t rst:1;
    u_int16_t psh:1;
    u_int16_t ack:1;
    u_int16_t urg:1;
    u_int16_t res2:2;
# elif __BYTE_ORDER == __BIG_ENDIAN
    u_int16_t doff:4;
    u_int16_t res1:4;
    u_int16_t res2:2;
    u_int16_t urg:1;
    u_int16_t ack:1;
    u_int16_t psh:1;
    u_int16_t rst:1;
    u_int16_t syn:1;
    u_int16_t fin:1;
# else

可以看到它是用系統(tǒng)定義的宏來區(qū)分大端還是小端迎罗。通過代碼跳轉(zhuǎn)可以找到__BYTE_ORDER宏的定義

 #define __BYTE_ORDER __LITTLE_ENDIAN    

或者纹安,在結(jié)構(gòu)體內(nèi)不使用位域砂豌,而是使用字節(jié)來存儲(比如8位),再通過位運算來計算出字節(jié)具體若干位對應(yīng)的值塔粒,比如剛才的Widget類可以改寫成下面這樣來避免機器大小端影響筐摘。

struct Widget
{
    uint8_t x;

    uint8_t a() const { return (x & 0xf0) >> 4; }
    uint8_t b() const { return (x & 0x0e) >> 1; }
    uint8_t c() const { return (x & 0x01); }

    void show()
    {
        printf("%d, %d, %d\n", a(), b(), c());
    }
};

void func()
{
    uint8_t x = 0b11100110;
    auto p = (Widget*)&x;
    p->show();  // output: 14, 3, 0
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市圃酵,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌薪韩,老刑警劉巖,帶你破解...
    沈念sama閱讀 221,548評論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件俘陷,死亡現(xiàn)場離奇詭異岭洲,居然都是意外死亡坎匿,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,497評論 3 399
  • 文/潘曉璐 我一進店門告私,熙熙樓的掌柜王于貴愁眉苦臉地迎上來承桥,“玉大人,你說我怎么就攤上這事蜀撑∈1颍” “怎么了?”我有些...
    開封第一講書人閱讀 167,990評論 0 360
  • 文/不壞的土叔 我叫張陵沃饶,是天一觀的道長轻黑。 經(jīng)常有香客問我,道長馆揉,這世上最難降的妖魔是什么抖拦? 我笑而不...
    開封第一講書人閱讀 59,618評論 1 296
  • 正文 為了忘掉前任勤讽,我火速辦了婚禮,結(jié)果婚禮上向臀,老公的妹妹穿的比我還像新娘。我一直安慰自己君纫,他們只是感情好芹彬,可當(dāng)我...
    茶點故事閱讀 68,618評論 6 397
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著会喝,像睡著了一般玩郊。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上译红,一...
    開封第一講書人閱讀 52,246評論 1 308
  • 那天侦厚,我揣著相機與錄音,去河邊找鬼诗宣。 笑死,一個胖子當(dāng)著我的面吹牛梧田,可吹牛的內(nèi)容都是我干的侧蘸。 我是一名探鬼主播,決...
    沈念sama閱讀 40,819評論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼穿稳,長吁一口氣:“原來是場噩夢啊……” “哼晌坤!你這毒婦竟也來了旦袋?” 一聲冷哼從身側(cè)響起它改,我...
    開封第一講書人閱讀 39,725評論 0 276
  • 序言:老撾萬榮一對情侶失蹤央拖,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后鲜戒,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,268評論 1 320
  • 正文 獨居荒郊野嶺守林人離奇死亡伦腐,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,356評論 3 340
  • 正文 我和宋清朗相戀三年柏蘑,在試婚紗的時候發(fā)現(xiàn)自己被綠了嗅剖。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 40,488評論 1 352
  • 序言:一個原本活蹦亂跳的男人離奇死亡黔攒,死狀恐怖强缘,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情赏胚,我是刑警寧澤,帶...
    沈念sama閱讀 36,181評論 5 350
  • 正文 年R本政府宣布觉阅,位于F島的核電站典勇,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏割笙。R本人自食惡果不足惜眯亦,卻給世界環(huán)境...
    茶點故事閱讀 41,862評論 3 333
  • 文/蒙蒙 一般码、第九天 我趴在偏房一處隱蔽的房頂上張望板祝。 院中可真熱鬧,春花似錦扔字、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,331評論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至琢蛤,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間套才,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,445評論 1 272
  • 我被黑心中介騙來泰國打工背伴, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留峰髓,地道東北人。 一個月前我還...
    沈念sama閱讀 48,897評論 3 376
  • 正文 我出身青樓疾掰,卻偏偏與公主長得像,于是被迫代替她去往敵國和親静檬。 傳聞我的和親對象是個殘疾皇子并级,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 45,500評論 2 359

推薦閱讀更多精彩內(nèi)容

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn)广恢,斷路器,智...
    卡卡羅2017閱讀 134,696評論 18 139
  • 字符編碼方案的演變與字節(jié)序 一钉迷、字符編碼方案的演變 1. 前文已經(jīng)提及,編號字符集CCS(簡稱字符集)與字符編碼方...
    笨笨阿林閱讀 1,140評論 1 3
  • __block和__weak修飾符的區(qū)別其實是挺明顯的:1.__block不管是ARC還是MRC模式下都可以使用糠聪,...
    LZM輪回閱讀 3,329評論 0 6
  • 1.寫一個NSString類的實現(xiàn) +(id)initWithCString:(c*****t char *)nu...
    韓七夏閱讀 3,772評論 2 37
  • 坦誠是我最贊賞的東西 尤其當(dāng)這種幻影出現(xiàn)在我的生命中 我是一個很好的傾聽者 一直以來都是 繁華歲月 一年來過得平淡...
    文墨糯糯閱讀 324評論 0 0