C和指針_第10章_結(jié)構(gòu)和聯(lián)合_學(xué)習(xí)筆記
1.結(jié)構(gòu)
struct?{
int?a;
char?b;
float?c;
}x;
struct?{
int?a;
char?b;
float?c;
}y[20],?*z;警告:以上兩個(gè)聲明被編譯器當(dāng)做兩個(gè)截然不同的類型。即y和z為同一類型,但與x類型不用。
? ? 使用結(jié)構(gòu)標(biāo)簽聲明結(jié)構(gòu):
struct?SIMPLE{
int?a;
char?b;
float?c;
};? ? 此時(shí)SIMPLE是結(jié)構(gòu)標(biāo)簽,使用標(biāo)簽創(chuàng)建需要如下代碼:
struct?SIMPLE?x; struct?SIMPLE?y[20],?*z;
? ? 使用typedef定義一個(gè)新類型:
typedef?struct?{
int?a;
char?b;
float?c;
}SIMPLE;? ? 此時(shí)SIMPLE是一個(gè)類型名,使用創(chuàng)建變量如下所示:
SIMPLE?x; SIMPLE?y[20],?*z;
1.5結(jié)構(gòu)自引用
? ? 結(jié)構(gòu)中不能包含類型為結(jié)構(gòu)本身的成員,但可以包含一個(gè)指向類型為結(jié)構(gòu)本身的成員的指針。注意:
//錯(cuò)誤
typedef?struct?{
int?a;
char?b;
float?c;
struct?SIMPLE?*d;
}SIMPLE;
//正確
struct?SIMPLE{
int?a;
char?b;
float?c;
struct?SIMPLE?*d;
};? ? 同時(shí)使用結(jié)構(gòu)標(biāo)簽和typedef也是正確的聲明方式:
typedef?struct?SIMPLE{
int?a;
char?b;
float?c;
struct?SIMPLE?*d;
}SIMPLE;3.結(jié)構(gòu)的內(nèi)存分配
? ? 編譯器按照成員列表的順序一個(gè)接一個(gè)的給每個(gè)成員分配內(nèi)存,但要滿足正確的編劇對齊要求。一般而言:
struct?SIMPL{
int?b;
char?a;
char?c;
};比下面聲明要占用更少的內(nèi)存:
struct?SIMPL{
char?a;
????????int?b;
char?c;
};? ? sizeof操作符可以獲得一個(gè)結(jié)構(gòu)的整體長度,包括因邊界對齊而跳過的字節(jié)。如果必須確定結(jié)構(gòu)某個(gè)成員的實(shí)際位置,應(yīng)該考慮邊界對齊因素,可以使用offsetof宏(定義于stddef.h)。
offsetof(?type,?member?);
? ? type是結(jié)構(gòu)類型,member是需要的成員名。
4.結(jié)構(gòu)做函數(shù)參數(shù)。
? ? 為提示效率,對較大的結(jié)構(gòu)體傳遞參數(shù)時(shí)采用指針會比傳遞值更優(yōu)效率。擔(dān)心傳遞指針會導(dǎo)致函數(shù)修改結(jié)構(gòu)體成員值可采用如下代碼:
void?func(?SIMPLE?const?*simple);
? ? 為提示效率還可以采用寄存器類型,即如下代碼:
void?func(?register?SIMPLE?const?*simple);
5.位段
? ? 位段的聲明和結(jié)構(gòu)體類似,但其的成員是一個(gè)或多個(gè)位的字段。這些不同長度的字段實(shí)際存儲于一個(gè)或多個(gè)整型變量中。
struct?CHAR?{
????unsigned?ch????????:?7;
????unsigned?font??????:?6;
????unsigned?size??????:?19;
};注意:
? ? 1.注重可移植性的程序應(yīng)避免使用位段。
? ? 2.位段中位的最大數(shù)目。聲明于32位機(jī)的位段可能在16位的機(jī)器上無法運(yùn)行。
? ? 3.位段中的成員在內(nèi)存中是從左往右分配還是從右往左分配。
? ? 4.當(dāng)一個(gè)聲明指定了兩個(gè)位段,第2個(gè)位段比較大,無法容納于第1個(gè)位段剩余位時(shí),編譯器可能把第2個(gè)位段放在內(nèi)存的下一個(gè)字,也可能直接放在第1個(gè)位段后面,從而在兩個(gè)內(nèi)存位置的邊界形成重疊。
6.聯(lián)合
? ? 聯(lián)合中的所有成員引用的是內(nèi)存中的相同位置,所占內(nèi)存空間為成員中占用內(nèi)存空間最大的成員所占的內(nèi)存空間。因此,若成員的占用內(nèi)存空間大小差距懸殊,采用在聯(lián)合中存儲指向不同成員的指針而不是直接存儲成員本身的方法可以節(jié)省內(nèi)存。聯(lián)合變量可以被初始化,但這個(gè)初始值必須是聯(lián)合第1個(gè)成員的類型,并且必須位于一對花括號內(nèi):
union?{
????int?a;
????float?b;
????char?c[4];
}?x?=?{?5?};????結(jié)構(gòu)和聯(lián)合的結(jié)合使用以節(jié)省內(nèi)存:
struct?VARIABLE?{
????enum?{?INT,?FLOAT,?STRING?}?type;
????union?{
????????int?i;
????????float?f;
????????char?*s;
????}value;
};




