C語言超詳細(xì)講解結(jié)構(gòu)體與聯(lián)合體的使用
結(jié)構(gòu)體
結(jié)構(gòu)體內(nèi)存對齊問題:
當(dāng)我們在計(jì)算結(jié)構(gòu)體的大小時,我們便需要清楚的知道結(jié)構(gòu)體內(nèi)存對齊是什么。
存在內(nèi)存對齊的原因可細(xì)分為兩個:
平臺原因:
不是所有的硬件平臺都能方位任意地址上的任意數(shù)據(jù);某些硬件平臺只能在某些地址處取某些特定類型的數(shù)據(jù),否則會拋出硬件異常。
性能原因:
首先內(nèi)存對齊可以提高程序的性能,當(dāng)訪問未對其的內(nèi)存空間時,有時候處理器需要進(jìn)行兩次訪問,而當(dāng)訪問對齊的內(nèi)存時,只需要一次就夠了。這同時也被叫做 用空間換取時間。
結(jié)構(gòu)體的對齊規(guī)則:
1.第一個成員在與結(jié)構(gòu)體變量偏移量為0的地址處。
2.其他成員變量要對齊到某各數(shù)字(對齊數(shù))的整數(shù)倍的地址處。
對齊數(shù)=編譯器默認(rèn)的一個對齊數(shù) 與 該成員大小的較小值。(VS中默認(rèn)的值為8)
3.結(jié)構(gòu)體總大小為最大對齊數(shù)(每個成員變量都有一個對齊數(shù))的整數(shù)倍。
4.如果嵌套了結(jié)構(gòu)體的情況,嵌套的結(jié)構(gòu)體對齊到自己的最大對齊數(shù)的整數(shù)倍處,結(jié)構(gòu)體的整體大小就是所有最大對齊數(shù)(含嵌套結(jié)構(gòu)體的對齊數(shù))的整數(shù)倍。
舉例1:
#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
int main()
{
struct s1
{
char c1; // 1字節(jié)
int i; // 4字節(jié)
char c2; // 1字節(jié)
};
printf("%d\n", sizeof(struct s1));
}輸出結(jié)果為:

解釋如下:
我們易知內(nèi)存會為結(jié)構(gòu)體開辟一塊空間來給結(jié)構(gòu)體存儲數(shù)據(jù),從而我們可以用下圖的方式將該空間給表示出來:

舉例2:
#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
int main()
{
struct s2
{
int i; // 4字節(jié)
char c1;// 1字節(jié)
char c2;// 1字節(jié)
};
printf("%d\n", sizeof(struct s2));
}輸出結(jié)果為:

解釋如下:

舉例3:
#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
struct s3
{
double d; // 8字節(jié)
char c; // 1字節(jié)
int i; // 4字節(jié)
};
struct s4
{
char c1; // 1字節(jié)
struct s3; // 16字節(jié)
double d; // 8字節(jié)
};
int main()
{
printf("%d\n", sizeof(struct s4));
return 0;
}輸出結(jié)果為:

解釋如下:

結(jié)論總結(jié):
當(dāng)我們想內(nèi)存對齊的同時也想節(jié)省空間時,可以將空間小的變量集中在一起??!
offsetof-宏
用途:計(jì)算結(jié)構(gòu)體成員相對于起始位置的偏移量的
注意:使用該函數(shù)時,應(yīng)該引用頭文件 #include <stddef.h>
舉例:
#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
#include<stddef.h>
struct s1
{
char c1;
int i;
char c2;
};
int main()
{
printf("%u\n", offsetof(struct s1,c1));
printf("%u\n", offsetof(struct s1, i));
printf("%u\n", offsetof(struct s1, c2));
}輸出結(jié)果為:

位段
位段的成員類型必須為: int、unsigned int、signed int
位段的空間是按照需要以4個字節(jié)(int)或者1個字節(jié)(char )的方式來開辟的
位段的成員名后邊有一個冒號和一個數(shù)字!
舉例1:
#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
struct s5
{ // 位段所代表的意思
int _a : 2; // _a 占 2個bit位
int _b : 5; // _b 占 5個bit位
int _c : 10;// _c 占 10個bit位
int _d : 30;// _d 占 30個bit位
};
int main()
{
printf("%d\n", sizeof(s5));
return 0;
}輸出結(jié)果為:( 原本的字節(jié)大小為 16 字節(jié) =16*8=128 bit 現(xiàn)在的字節(jié)大小為 8字節(jié)且只占 2+5+10+30 = 47bite)

那為啥不是占用7字節(jié)呢?7字節(jié)有 7*8=56bite 也夠使用?。?/p>
我們便需要根據(jù)位段的規(guī)定來解釋,當(dāng)為(int)類型時內(nèi)存空間每次都是以4字節(jié)的大小來開辟空間的,當(dāng)為(char)類型時內(nèi)存空間每次都是以1字節(jié)的大小來開辟空間的,所給例子為int類型,當(dāng)所定的第一個4字節(jié)空間不夠用時,便會再開辟一塊大小為4字節(jié)的空間來供其存儲,從而輸出結(jié)果為8字節(jié)。
結(jié)論:
我們可以根據(jù)所定義的整型數(shù)字大小,利用位段來給其分配相適應(yīng)大小的空間,從而有效的幫我們進(jìn)行內(nèi)存空間的節(jié)省。
舉例2:
#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
struct S
{
char a : 3;
char b : 4;
char c : 5;
char d : 4;
};
int main()
{
struct S s = { 0 };
s.a = 10;
s.b = 12;
s.c = 3;
s.d = 4;
return 0;
}我們想要知道位段在(VS編譯器)內(nèi)存中是如何存儲的,便可以打開監(jiān)控來進(jìn)行調(diào)試
如圖所示:
(結(jié)構(gòu)體變量s 剛開始初始化了 3字節(jié))

(結(jié)構(gòu)體變量s 經(jīng)過賦值之后存儲數(shù)值的變化)

解釋如下圖:

位段的跨平臺問題:
1. int 位段被當(dāng)成有符號數(shù)還是無符號數(shù)是不確定的。
2. 位段中最大位的數(shù)目不能確定。(16位機(jī)器最大16,32位機(jī)器最大32),寫成27,在16位機(jī)器會出問題。
3.位段中的成員在內(nèi)存中從左向右分配,還是從右向左分配標(biāo)準(zhǔn)尚未定義。
4. 當(dāng)一個結(jié)構(gòu)包含兩個位段,第二個位段成員比較大,無法容納于第一個位段剩余的位時,是舍棄剩余的位還是利用,這是不確定的。
結(jié)論:
位段涉及很多不確定因素,位段是不垮平臺的,注重可移植的程序應(yīng)該避免使用位段。
枚舉
枚舉類型是某類數(shù)據(jù)可能取值的集合
例子:一周內(nèi)星期的取值為7天,可以一一列舉出來
定義方式及使用方式:
#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
enum day
{
Mon,
Tues,
Wed,
Thur,
Fri,
Sat,
Sun
};
int main()
{
enum day s1 = Mon;
enum day s2 = Sat;
return 0;
}舉例1:
#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
enum color
{
blue,
green,
yellow
};
int main()
{
printf("%d\n", blue);
printf("%d\n", green);
printf("%d\n", yellow);
return 0;
}輸出結(jié)果為: (由結(jié)果知枚舉常量會被自動從0開始一次往下賦值)

拓展:
#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
enum color
{
blue=4,
green=7,
yellow
};
int main()
{
printf("%d\n", blue);
printf("%d\n", green);
printf("%d\n", yellow);
return 0;
}輸出結(jié)果為:(由此可知枚舉常量我們可以自定義賦值,未賦值常量為其上一常量的值+1)

#define 也可以用來定義常量,那用枚舉來定義常量的優(yōu)點(diǎn)為:
1.增加代碼的可讀性和可維護(hù)性
2. 和#define定義的標(biāo)識符比較枚舉有類型檢查,更加嚴(yán)謹(jǐn)
3.防止了命名污染(封裝)
4. 便于調(diào)試
5. 使用方便,一次可以定義多個常量
聯(lián)合體(共用體)
特點(diǎn):
各成員共享一段內(nèi)存空間,一個 聯(lián)合變量的長度等于各成員中最長的長度。
舉例1:
#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
union Un
{
char c;
int i;
};
int main()
{
union Un u1;
printf("%d\n", sizeof(u1));
printf("%p\n", &u1);
printf("%p\n", &u1.c);
printf("%p\n", &u1.i);
}輸出結(jié)果為:

如圖所示:

應(yīng)用:(用來判斷編譯器是大端存儲還是小端存儲)
#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
int chek_sys()
{
union Un //創(chuàng)建一個聯(lián)合體 char c 和 int i 共用同一塊存儲空間
{
char c;
int i;
}u;
u.i = 1; // 這里給i賦值為1
//若為小端存儲時內(nèi)存中所存儲:01 00 00 00(16進(jìn)制) 為大端存儲: 00 00 00 01(16進(jìn)制)
return u.c; //這里直接返回 char c 與 int i 所共用空間的值
//返回1字節(jié)大小的值 即 int i 以16進(jìn)制方式所存儲的前兩位數(shù)字 ,若值為1則為小端 若值為0則為大端
}
int main()
{
if (1 == chek_sys())
{
printf("小端\n");
}
else
{
printf("大端\n");
}
return 0;
}輸出結(jié)果:

舉例2:(計(jì)算聯(lián)合體的大?。?/p>
#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
union Un
{
char arr[5];//5字節(jié)
int i; //4字節(jié)
};
int main()
{
printf("%d\n", sizeof(union Un));
return 0;
}輸出結(jié)果為:(我們從結(jié)果得知聯(lián)合體也存在內(nèi)存對齊)

解釋:
char類型數(shù)組先占用5字節(jié),其對齊數(shù)為1字節(jié)(char),int i占用4字節(jié)與char類型數(shù)組公用同一塊空間,其對齊數(shù)為4字節(jié)(int),該聯(lián)合體所占5字節(jié),但存在內(nèi)存對齊,需為4字節(jié)的倍數(shù),從而要浪費(fèi)3字節(jié)空間使其為8字節(jié)。
以上便是關(guān)于結(jié)構(gòu)體和聯(lián)合體的全部內(nèi)容 !
如有錯誤或者能改進(jìn)的地方 請各大佬指出 我會及時改正??!
到此這篇關(guān)于C語言超詳細(xì)講解結(jié)構(gòu)體與聯(lián)合體的使用的文章就介紹到這了,更多相關(guān)C語言結(jié)構(gòu)體與聯(lián)合體內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
MFC中動態(tài)創(chuàng)建控件以及事件響應(yīng)實(shí)現(xiàn)方法
這篇文章主要介紹了MFC中動態(tài)創(chuàng)建控件以及事件響應(yīng)實(shí)現(xiàn)方法,詳細(xì)講解了MFC中動態(tài)創(chuàng)建控件以及事件響應(yīng)的概念與實(shí)現(xiàn)方法,具有一定的實(shí)用價(jià)值,需要的朋友可以參考下2014-10-10
C語言中如何實(shí)現(xiàn)單鏈表刪除指定結(jié)點(diǎn)
這篇文章主要介紹了C語言中如何實(shí)現(xiàn)單鏈表刪除指定結(jié)點(diǎn),具有很好的參考價(jià)值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-07-07
創(chuàng)建二叉樹 二叉樹如何刪除節(jié)點(diǎn)操作教程
本文將詳細(xì)介紹二叉樹的創(chuàng)建,節(jié)點(diǎn)刪除,節(jié)點(diǎn)增加等一系列操作方法,需要的朋友可以參考下2012-12-12
C++使用QTcreator創(chuàng)建動態(tài)庫流程
在工程中,經(jīng)常會根據(jù)不同的場景需求將類封裝成庫文件,本文主要介紹了C++使用QTcreator創(chuàng)建動態(tài)庫流程,具有一定的參考價(jià)值,感興趣的可以了解一下2024-06-06
C/C++字符串與數(shù)字互轉(zhuǎn)的實(shí)現(xiàn)
這篇文章主要介紹了C/C++字符串與數(shù)字互轉(zhuǎn)的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-01-01
VS2010 boost標(biāo)準(zhǔn)庫開發(fā)環(huán)境安裝教程
這篇文章主要為大家詳細(xì)介紹了VS2010 boost標(biāo)準(zhǔn)庫開發(fā)環(huán)境的安裝教程,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-04-04

