當前位置:才華齋>計算機>C語言>

C語言段位操作知識整理

C語言 閱讀(2.93W)

C語言的應用範圍廣泛,具備很強的資料處理能力,不僅僅是在軟體開發上,而且各類科研都需要用到C語言,適於編寫系統軟體,三維,二維圖形和動畫,具體應用比如微控制器以及嵌入式系統開發。今天,小編為大家搜尋整理了C語言段位操作知識整理,希望大家能有所收穫,更多精彩內容請持續關注我們應屆畢業生考試網!

C語言段位操作知識整理

  1.位段結構中位段的定義格式為:

unsigned <成員名>:<二進位制位數>

例如:

struct bytedata

{unsigned a:2; /*位段a,佔2位*/

unsigned:6; /*無名位段,佔6位,但不能訪問*/

unsigned:0; /*無名位段,佔0位,表下一位段從下一字邊界開始*/

unsigned b:10; /*位段b,佔10位*/

int i; /*成員i,從下一字邊界開始*/

}data;

2.

(1)一個位段必須儲存在同一儲存單元(即字)之中,不能跨兩個單元。如果其單元空間不夠,則剩餘空間不用,從下一個單元起存放該位段。

(2)可以通過定義長度為0的位段的方式使下一位段從下一儲存單元開始。

(3)可以定義無名位段。

(4)位段的長度不能大於儲存單元的長度。

(5)位段無地址,不能對位段進行取地址運算。

(6)位段可以以%d,%o,%x格式輸出。

(7)位段若出現在表示式中,將被系統自動轉換成整數。

拷貝2:C語言中的結構是有實現位段的能力的,噢!你問它到底是什麼形式是吧?這個問題呆會給你答案。讓我們先看看位段的作用:位段是在欄位的聲明後面加一個冒號以及一個表示欄位位長的整數來實現的。這種用法又被就叫作“深入邏輯元件的程式設計”,如果你對系統程式設計感興趣,那麼這篇文章你就不應該錯過!

我把使用位段的幾個理由告訴大家:1、它能把長度為奇數的資料包裝在一起,從而節省儲存的空間;2、它可以很方便地訪問一個整型值的部分內容。

首先我要提醒大家注意幾點:1、位段成員只有三種類型:int ,unsigned int 和signed int這三種(當然了,int型位段是不是可以取負數不是我說了算的,因為這是和你的'編譯器來決定的。位段,位段,它是用來表示欄位位長(bit)的,它只有整型值,不會有7.2這種float型別的,如果你說有,那你就等於承認了有7.2個人這個概念,當然也沒有char這個型別的);2、成員名後面的一個冒號和一個整數,這個整數指定該位段的位長(bit);3、許多編譯器把位段成員的字長限制在一個int的長度範圍之內;4、位段成員在記憶體的實現是從左到右還是從右到左是由編譯器來決定的,但二者皆對。

下面我們就來看看,它到底是什麼東西(我先假定大家的機器字長為32位):

Struct WORD

{

unsigned int chara: 6:

unsigned int font : 7;

unsigned int maxsize : 19;

};

Struct WORD chone;

這一段是從我編寫的一個文字格式化軟體摘下來的,它最多可以容納64(既我說的unsigned int chara :6; 它總共是6位)個不同的字元值,可以處理128(既unsigned int font : 7 ;既2的7次方)種不同的字型,和2的19次方的單位長度的字。大家都可以看到maxsize是19位,它是無法被一個short int 型別的值所容納的,我們又可以看到其餘的成員的長度比char還小,這就讓我們想起讓他們共享32位機器字長,這就避免用一個32位的整數來表示maxsize的位段。怎麼樣?還要注意的是剛才的那一段程式碼在16位字長的機器上是無法實現的,為什麼?提醒你一下,看看上面提醒的第3點,你會明白的!

你是不是發現這個東西沒有用啊?如果你點頭了,那你就錯了!這麼偉大的創造怎麼會沒有用呢(你對系統程式設計不感興趣,相信你會改變這麼一個觀點的)?磁碟控制器大家應該知道吧?軟碟機與它的通訊我們來看看是怎麼實現的下面是一個磁碟控制器的暫存器:

│←5→│←5→│←9→│←8→│←1→│←1→∣←1→∣←1→∣←1→∣

上面位段從左到右依次代表的含義為:5位的命令,5位的扇區,9位的磁軌,8位的錯誤程式碼,1位的HEAD LOADED,1位的防寫,1位的DISK SPINNING,1位的錯誤判斷符,還有1位的READY位。它要怎麼來實現呢?你先自己寫寫看:

struct DISK_FORMAT

{

unsigned int command : 5;

unsigned sector : 5;

unsigned track : 9 ;

unsigned err_code : 8;

unsigned ishead_loaded : 1;

unsigned iswrit_protect : 1;

unsigned isdisk_spinning : 1;

unsigned iserr_ocur : 1;

undigned isready :1 ;

};

注:程式碼中除了第一行使用了unsigned int 來宣告位段後就省去了int ,這是可行的,詳見ANCI C標準。

如果我們要對044c18bfH的地址進行訪問的話,那就這樣:

#define DISK ((struct DISK_FORMAT *)0x044c18bf)

DISK->sector=fst_sector;

DISK->track=fst_track;

DISK->command=WRITE;

當然那些都是要巨集定義的哦!

我們用位段來實現這一目的是很方便的,其實這也可以用移位或遮蔽來實現,你嘗試過就知道哪個更方便了!

測試程式碼:

#i nclude

int main()

{

struct

{

unsigned short s1 : 4;

unsigned short s2 : 3;

unsigned short s3 : 2;

}x;

char c= 0x7A; // 0111 1010 b

x.s1 = c;

printf( "%x/n", x.s1 );

return 0;

}

根據編譯器的不同,可能出現大端和小端的問題,小端就是從低位開始取值,大端就是從高位取值。

常見為小端模式。