国产精品久久久aaaa,日日干夜夜操天天插,亚洲乱熟女香蕉一区二区三区少妇,99精品国产高清一区二区三区,国产成人精品一区二区色戒,久久久国产精品成人免费,亚洲精品毛片久久久久,99久久婷婷国产综合精品电影,国产一区二区三区任你鲁

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

C語言中的位域是什么?

GReq_mcu168 ? 來源:李逍遙 ? 作者:李逍遙 ? 2021-01-13 16:23 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

位域是什么?

有些數(shù)據(jù)在存儲時并不需要占用一個完整的字節(jié),只需要占用一個或幾個二進制位即可。例如開關只有通電和斷電兩種狀態(tài),用 0 和 1 表示足以,也就是用一個二進位。正是基于這種考慮,C語言又提供了一種叫做位域的數(shù)據(jù)結構。 在結構體定義時,我們可以指定某個成員變量所占用的二進制位數(shù)(Bit),這就是位域。請看下面的例子:

1structbs{ 2unsignedm; 3unsignedn:4; 4unsignedcharch:6; 5}; :后面的數(shù)字用來限定成員變量占用的位數(shù)。成員 m 沒有限制,根據(jù)數(shù)據(jù)類型即可推算出它占用 4 個字節(jié)(Byte)的內存。成員 n、ch 被:后面的數(shù)字限制,不能再根據(jù)數(shù)據(jù)類型計算長度,它們分別占用 4、6 位(Bit)的內存。 n、ch 的取值范圍非常有限,數(shù)據(jù)稍微大些就會發(fā)生溢出,請看下面的例子:1#include 2intmain(){ 3structbs{ 4unsignedm; 5unsignedn:4; 6unsignedcharch:6; 7}a={0xad,0xE,'$'}; 8//第一次輸出 9printf("%#x,%#x,%c ",a.m,a.n,a.ch); 10//更改值后再次輸出 11a.m=0xb8901c; 12a.n=0x2d; 13a.ch='z'; 14printf("%#x,%#x,%c ",a.m,a.n,a.ch); 15system("pause"); 16return0; 17} 運行結果:

對于 n 和 ch,第一次輸出的數(shù)據(jù)是完整的,第二次輸出的數(shù)據(jù)是殘缺的。 第一次輸出時,n、ch 的值分別是 0xE、0x24('$' 對應的 ASCII 碼為 0x24),換算成二進制是1110、10 0100,都沒有超出限定的位數(shù),能夠正常輸出。 第二次輸出時,n、ch 的值變?yōu)?0x2d、0x7a('z' 對應的 ASCII 碼為 0x7a),換算成二進制分別是10 1101、111 1010,都超出了限定的位數(shù)。超出部分被直接截去,剩下1101、11 1010,換算成十六進制為 0xd、0x3a(0x3a 對應的字符是 :)。

C語言標準規(guī)定,位域的寬度不能超過它所依附的數(shù)據(jù)類型的長度。通俗地講,成員變量都是有類型的,這個類型限制了成員變量的最大長度,:后面的數(shù)字不能超過這個長度。

例如上面的 bs,n 的類型是 unsigned int,長度為 4 個字節(jié),共計 32 位,那么 n 后面的數(shù)字就不能超過 32;ch 的類型是 unsigned char,長度為 1 個字節(jié),共計 8 位,那么 ch 后面的數(shù)字就不能超過 8。 我們可以這樣認為,位域技術就是在成員變量所占用的內存中選出一部分位寬來存儲數(shù)據(jù)。

C語言標準還規(guī)定,只有有限的幾種數(shù)據(jù)類型可以用于位域。在 ANSI C 中,這幾種數(shù)據(jù)類型是 int、signed int 和 unsigned int(int 默認就是 signed int);到了 C99,_Bool 也被支持了。

但編譯器在具體實現(xiàn)時都進行了擴展,額外支持了 char、signed char、unsigned char 以及 enum 類型,所以上面的代碼雖然不符合C語言標準,但它依然能夠被編譯器支持。

位域的存儲

C語言標準并沒有規(guī)定位域的具體存儲方式,不同的編譯器有不同的實現(xiàn),但它們都盡量壓縮存儲空間。位域的具體存儲規(guī)則如下:

當相鄰成員的類型相同時,如果它們的位寬之和小于類型的 sizeof 大小,那么后面的成員緊鄰前一個成員存儲,直到不能容納為止;如果它們的位寬之和大于類型的 sizeof 大小,那么后面的成員將從新的存儲單元開始,其偏移量為類型大小的整數(shù)倍。

以下面的位域 bs 為例:

1#include 2intmain(){ 3structbs{ 4unsignedm:6; 5unsignedn:12; 6unsignedp:4; 7}; 8printf("%d ",sizeof(structbs)); 9return0; 10} 運行結果:

m、n、p 的類型都是 unsigned int,sizeof 的結果為 4 個字節(jié)(Byte),也即 32 個位(Bit)。m、n、p 的位寬之和為6+12+4 = 22,小于 32,所以它們會挨著存儲,中間沒有縫隙。 sizeof(struct bs)的大小之所以為 4,而不是 3,是因為要將內存對齊到 4 個字節(jié),以便提高存取效率。 如果將成員 m 的位寬改為 22,那么輸出結果將會是 8,因為22+12 = 34,大于 32,n 會從新的位置開始存儲,相對 m 的偏移量是sizeof(unsigned int),也即 4 個字節(jié)。 如果再將成員 p 的位寬也改為 22,那么輸出結果將會是 12,三個成員都不會挨著存儲。

當相鄰成員的類型不同時,不同的編譯器有不同的實現(xiàn)方案,GCC 會壓縮存儲,而 VC/VS 不會。

請看下面的位域 bs:

1#include 2intmain(){ 3structbs{ 4unsignedm:12; 5unsignedcharch:4; 6unsignedp:4; 7}; 8printf("%d ",sizeof(structbs)); 9return0; 10} 在 GCC 下的運行結果為 4,三個成員挨著存儲;在 VC/VS 下的運行結果為 12,三個成員按照各自的類型存儲(與不指定位寬時的存儲方式相同)。 m 、ch、p 的長度分別是 4、1、4 個字節(jié),共計占用 9 個字節(jié)內存,為什么在 VC/VS 下的輸出結果卻是 12 呢?期待您的回復。

如果成員之間穿插著非位域成員,那么不會進行壓縮。例如對于下面的 bs:

1structbs{ 2unsignedm:12; 3unsignedch; 4unsignedp:4; 5}; 在各個編譯器下 sizeof 的結果都是 12。 通過上面的分析,我們發(fā)現(xiàn)位域成員往往不占用完整的字節(jié),有時候也不處于字節(jié)的開頭位置,因此使用&獲取位域成員的地址是沒有意義的,C語言也禁止這樣做。地址是字節(jié)(Byte)的編號,而不是位(Bit)的編號。

無名位域

位域成員可以沒有名稱,只給出數(shù)據(jù)類型和位寬,如下所示:

1structbs{ 2intm:12; 3int:20;//該位域成員不能使用 4intn:4; 5}; 無名位域一般用來作填充或者調整成員位置。因為沒有名稱,無名位域不能使用。 上面的例子中,如果沒有位寬為 20 的無名成員,m、n 將會挨著存儲,sizeof(struct bs)的結果為 4;有了這 20 位作為填充,m、n 將分開存儲,sizeof(struct bs)的結果為 8。

原文標題:C語言位域(位段)詳解

文章出處:【微信公眾號:玩轉單片機】歡迎添加關注!文章轉載請注明出處。

責任編輯:haq

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權轉載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 存儲
    +關注

    關注

    13

    文章

    4787

    瀏覽量

    90057
  • C語言
    +關注

    關注

    183

    文章

    7644

    瀏覽量

    145571

原文標題:C語言位域(位段)詳解

文章出處:【微信號:mcu168,微信公眾號:硬件攻城獅】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

    評論

    相關推薦
    熱點推薦

    c語言中的代碼優(yōu)化

    能夠提升算法效率。由于乘除運算指令周期一般比移位運算大。C語言運算除了能夠提升運算效率外,在嵌入式系統(tǒng)的編程中,它的另外一 個最典型的應用,并且十分普遍地正在被使用著的是間的與(
    發(fā)表于 01-12 09:45

    請問C語言中整形溢出的解決方法有哪些?

    C語言中整形溢出的解決方法有哪些?
    發(fā)表于 12-29 07:33

    單片機c語言中定義字節(jié)怎么定義?

    定義可以用***it但是定義字節(jié)呢?還有就是比如匯編中的DPTR的DPH和DPL在c語言中怎樣定義的呢?
    發(fā)表于 12-29 06:32

    請問C語言中整形溢出會產(chǎn)生哪些異常行為?

    C語言中整形溢出會產(chǎn)生哪些異常行為?
    發(fā)表于 12-26 07:05

    請問C語言中整形溢出對哪些應用場景影響較大?

    C語言中整形溢出對哪些應用場景影響較大
    發(fā)表于 12-24 08:24

    C語言C++的區(qū)別及聯(lián)系

    創(chuàng)建源文件時什么都不給,默認是.cpp。 3、返回值 C語言中,如果一個函數(shù)沒有指定返回值類型,默認返回int類型;C++中,如果一個函數(shù)沒有返回值則必須指定為void。 4、參數(shù)列表 在
    發(fā)表于 12-24 07:23

    Typedef在C語言中的應用

    Typedef 在C語言中頻繁用以聲明一個已經(jīng)存在的數(shù)據(jù)類型的同義字。也可以用預處理器做類似的事。例如,思考一下下面的例子: #define dPS struct s * typedef
    發(fā)表于 12-22 13:53

    C語言中可以嵌套匯編語言嗎?

    請問C語言中可以嵌套匯編語言嗎?
    發(fā)表于 12-15 08:14

    什么是

    有些信息在存儲時,并不需要占用一個完整的字節(jié), 而只需占幾個或一個二進制。 例如在存放一個開關量時,只有0和1 兩種狀態(tài), 用一二進位即可。為了節(jié)省存儲空間,并使處理簡便,C語言
    發(fā)表于 12-15 08:07

    C語言反轉的應用

    反轉(Bit Reversal) unsigned int reverse_bits(unsigned int num) { unsigned int numOfBits = sizeof
    發(fā)表于 12-12 06:42

    解讀

    只用1個bit就能夠放完,而一個整型卻是4個字節(jié),也就是32bit。這就造成了內存的浪費。 好在,C語言為我們提供了一種數(shù)據(jù)結構,稱為「」(也叫
    發(fā)表于 12-05 06:45

    C語言的分支結構介紹

    1.簡單if語句 C語言中的分支結構語句中的if條件語句。 簡單if語句的基本結構如下: 代碼語言:javascript if(表達式) { 執(zhí)行代碼塊; } 其語義是:如果表達式的值為真,則執(zhí)行其后的語句,否則不執(zhí)
    發(fā)表于 11-25 07:48

    C語言的常量介紹

    、-13; 實型常量:13.33、-24.4; 字符常量:‘a(chǎn)’、‘M’ 字符串常量:”I love china!” 在C語言中,可以用一個標識符來表示一個常量,稱之為符號常量。符號常量在使用之前必須先
    發(fā)表于 11-24 07:12

    Windows環(huán)境下32匯編語言中文資料

    電子發(fā)燒友網(wǎng)站提供《Windows環(huán)境下32匯編語言中文資料.rar》資料免費下載
    發(fā)表于 06-30 15:14 ?0次下載

    深入理解C語言C語言循環(huán)控制

    改變程序的執(zhí)行流程,使代碼更加靈活和可控。本文將詳細介紹這些語句的作用及其應用場景,并通過示例代碼進行說明。Part.1break語句C語言中break語句有兩種
    的頭像 發(fā)表于 04-29 18:49 ?2043次閱讀
    深入理解<b class='flag-5'>C</b><b class='flag-5'>語言</b>:<b class='flag-5'>C</b><b class='flag-5'>語言</b>循環(huán)控制