在 C 語言中我們經(jīng)常會使用到 struct和 union,那么它們兩個各自有何特點(diǎn)呢?今天我們就一探究竟。
創(chuàng)新互聯(lián)專業(yè)提供成都主機(jī)托管四川主機(jī)托管成都服務(wù)器托管四川服務(wù)器托管,支持按月付款!我們的承諾:貴族品質(zhì)、平民價(jià)格,機(jī)房位于中國電信/網(wǎng)通/移動機(jī)房,雅安電信機(jī)房服務(wù)有保障!
我們先來介紹下 struct。它可以看做是變量的集合,那么一個空的結(jié)構(gòu)體占多大內(nèi)存呢?這是一個有趣的問題,按照理論分析,它應(yīng)該是0。但是按照 C 語言的設(shè)計(jì)思想來說,不可能存在空結(jié)構(gòu)體的,定義一個空結(jié)構(gòu)體沒意義啊,所以應(yīng)該報(bào)錯的。下來我們就分別在 gcc 和 BCC 編譯器上實(shí)驗(yàn)下。由于代碼比較簡單,就不貼代碼了,我們直接來結(jié)果。圖一為在 gcc 編譯器下編譯的,圖二為在 BCC 編譯器下編譯的。
圖一
圖二
那么我們可以看到在 gcc 編譯器中,它支持我們的第一種看法,即認(rèn)為占0個字節(jié)的內(nèi)存。但是在 BCC 編譯器中,它認(rèn)為這樣是不合法的,定義空結(jié)構(gòu)體根本沒必要,所以直接報(bào)錯了。
那么我們在 C 語言中定義一個組數(shù)時(shí),平常情況下只能定義大小是固定的數(shù)組。有沒有什么辦法讓我們在 C 語言中定義一個動態(tài)大小的數(shù)組呢?辦法當(dāng)然是有的,這時(shí)我們就要用到我們的 struct了。我們可以利用 struct來定義一個大小待定的數(shù)組,我們稱之為柔性數(shù)組。 在 C 語言中結(jié)構(gòu)體的最后一個元素可以是大小未知的數(shù)組,那么在結(jié)構(gòu)體中的數(shù)組便是一個待使用的標(biāo)識符,并不占用存儲空間。不信嗎?我們來做個實(shí)驗(yàn),代碼如下:
#includestruct TS { int len; int array[]; }; int main() { printf("sizeof(struct TS) = %d\n", sizeof(struct TS)); return 0; }
我們先來分析下這個代碼,結(jié)構(gòu)體 TS 中定義了一個 int 類型的變量 len,還有個大小未知的數(shù)組 array。那么這個可以編譯通過嗎?如果可以,它的大小又會是多少呢?我們來看看結(jié)果:
我們可以看到編譯器并沒有報(bào)錯,也就證明是可以這樣定義的,并且它的大小為 4 。這說明數(shù)組 array 并沒有占用內(nèi)存。下來我們來介紹下柔性數(shù)組的用法,如下圖所示
我們來使用下柔性數(shù)組,代碼如下:
#include#include struct SoftArray { int len; int array[]; }; struct SoftArray* create_soft_array(int size) { struct SoftArray* ret = NULL; if( size > 0 ) { ret = (struct SoftArray*)malloc(sizeof(struct SoftArray) + sizeof(int) * size); ret->len = size; } return ret; } void delete_SoftArray(struct SoftArray* sa) { free(sa); } void func(struct SoftArray* sa) { int i = 0; if( NULL != sa ) { for(i=0; i len; i++) { sa->array[i] = i + 1; } } } int main() { int i = 0; struct SoftArray* sa = create_soft_array(5); func(sa); for(i=0; i len; i++) { printf("sa[%d] = %d\n", i, sa->array[i]); } delete_SoftArray(sa); return 0; }
我們來看下編譯后的結(jié)果
我們已經(jīng)成功實(shí)現(xiàn)了一個柔性數(shù)組,可以自己指定這個數(shù)組的大小了。
下來我們來介紹下 C 語言中的 union,它在語法上跟 struct 很像。但是 union 只分配最大的成員變量的空間,所有成員共享這個空間。union 的使用受系統(tǒng)大小端的影響,我們來看看系統(tǒng)的大小端內(nèi)存是怎樣分配的,如下圖所示:
那么在小端模式下,數(shù)據(jù)存儲在低位地址上。大端則相反,但是我們的程序取數(shù)據(jù)總是從低地址開始取的。在上圖中的程序中,如果系統(tǒng)是小端,則輸出為 1,反之則為 0。根據(jù) union 這個特性,我們可以寫一個判斷系統(tǒng)大小端的函數(shù)。這道題也是筆試中我們經(jīng)常會見到的,代碼如下:
#includeint System_mode() { union SM { int i; char c; }; union SM sm; sm.i = 1; return sm.c; } int main() { if( 1 == System_mode() ) { printf("小端模式\n"); } else { printf("大端模式\n"); } return 0; }
我們編譯后結(jié)果如下:
那么我們本節(jié)學(xué)習(xí)了 struct和 union 的有關(guān)特性,通過本節(jié)學(xué)習(xí),總結(jié)如下:1、struct中每個數(shù)據(jù)成員有獨(dú)立的存儲空間,可以通過最后的數(shù)組標(biāo)識符產(chǎn)生柔性數(shù)組;2、union中所有的數(shù)據(jù)成員共享同一個存儲空間,同時(shí)它的使用會受到系統(tǒng)大小端的影響。后面我們會繼續(xù)對 C 語言的學(xué)習(xí)。
歡迎大家一起來學(xué)習(xí) C 語言,可以加我QQ:243343083。