深入sizeof的使用詳解
更新時間:2013年05月27日 10:08:28 作者:
本篇文章是對sizeof的使用進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下
經(jīng)常在C下編程,不了解精通sizeof就有點(diǎn)說不過去了,本文可沒有sizeof與strlen的區(qū)別哦,盡管這個對比很經(jīng)典,不過,將就點(diǎn)吧。
一、sizeof的概念
sizeof是C語言的一種單目操作符,如C語言的其他操作符++、--等。它并不是函數(shù)。sizeof操作符以字節(jié)形式給出了其操作數(shù)的存儲大小。
操作數(shù)可以是一個表達(dá)式或括在括號內(nèi)的類型名。操作數(shù)的存儲大小由操作數(shù)的類型決定。
二、sizeof的使用方法
1、用于數(shù)據(jù)類型
sizeof使用形式:sizeof(type)
數(shù)據(jù)類型必須用括號括住。如sizeof(int),還有sizeof int ,或sizeof(variable)三種方式?!?br>
2、用于變量
sizeof使用形式:sizeof(var_name)或sizeof var_name
變量名可以不用括號括住。如sizeof (var_name),sizeof var_name等都是正確形式。帶括號的用法更普遍,大多數(shù)程序員采用這種形式?!?BR> 注意:sizeof操作符不能用于函數(shù)類型,不完全類型或位字段。不完全類型指具有未知存儲大小的數(shù)據(jù)類型,如未知存儲大小的數(shù)組類型、未知內(nèi)容的結(jié)構(gòu)或聯(lián)合類型、void類型等?!?BR> 如sizeof(max)若此時變量max定義為int max(),sizeof(char_v) 若此時char_v定義為char char_v [MAX]且MAX未知,sizeof(void)都不是正確形式。
三、sizeof的結(jié)果
sizeof操作符的結(jié)果類型是size_t,它在頭文件中typedef為unsigned int類型。該類型保證能容納實(shí)現(xiàn)所建立的最大對象的字節(jié)大小?!?BR> 1、若操作數(shù)具有類型char、unsigned char或signed char,其結(jié)果等于1?!?BR> ANSI C正式規(guī)定字符類型為1字節(jié)。
2、int、unsigned int 、short int、unsigned short 、long int 、unsigned long 、float、double、long double類型的sizeof
在ANSI C中沒有具體規(guī)定,大小依賴于實(shí)現(xiàn),一般可能分別為2、2、2、2、4、4、4、8、10?!?BR> 3、當(dāng)操作數(shù)是指針時,sizeof依賴于編譯器。例如Microsoft C/C++7.0中,near類指針字節(jié)數(shù)為2,far、huge類指針字節(jié)數(shù)為4。一般Unix的指針字節(jié)數(shù)為4。
4、當(dāng)操作數(shù)具有數(shù)組類型時,其結(jié)果是數(shù)組的總字節(jié)數(shù)。
5、聯(lián)合類型操作數(shù)的sizeof是其最大字節(jié)成員的字節(jié)數(shù)。結(jié)構(gòu)類型操作數(shù)的sizeof是這種類型對象的總字節(jié)數(shù),包括任何墊補(bǔ)在內(nèi)。
讓我們看如下結(jié)構(gòu):
struct A{
char b;
double x;
} a;
在某些機(jī)器上sizeof(a)=12,而一般sizeof(char)+ sizeof(double)=9?!?BR> 這是因?yàn)榫幾g器在考慮對齊問題時,在結(jié)構(gòu)中插入空位以控制各成員對象的地址對齊。如double類型的結(jié)構(gòu)成員x要放在被4整除的地址?!?BR> 6、如果操作數(shù)是函數(shù)中的數(shù)組形參或函數(shù)類型的形參,sizeof給出其指針的大小?!?br>
四、sizeof與其他操作符的關(guān)系
sizeof的優(yōu)先級(各操作符的優(yōu)先級可參考C語言運(yùn)算符優(yōu)先級列表(超詳細(xì)):)為2級,比/、%等3級運(yùn)算符優(yōu)先級高。它可以與其他操作符一起組成表達(dá)式。如i*sizeof(int);其中i為int類型變量?!?br>
五、sizeof的主要用途
1、sizeof操作符的一個主要用途是與存儲分配和I/O系統(tǒng)那樣的例程進(jìn)行通信。例如:
void *malloc(size_t size),
size_t fread(void *ptr, size_t size, size_t nmemb,FILE * stream)?!?BR> 2、sizeof的另一個的主要用途是計(jì)算數(shù)組中元素的個數(shù)。例如:
void * memset(void *s,int c,sizeof(s))?!?br>
六、建議
由于操作數(shù)的字節(jié)數(shù)在實(shí)現(xiàn)時可能出現(xiàn)變化,建議在涉及到操作數(shù)字節(jié)大小時用sizeof來代替常量計(jì)算。
本文主要包括二個部分,第一部分重點(diǎn)介紹在VC中,怎么樣采用sizeof來求結(jié)構(gòu)的大小,以及容易出現(xiàn)的問題,并給出解決問題的方法,
第二部分總結(jié)出VC中sizeof的主要用法。
1、 sizeof應(yīng)用在結(jié)構(gòu)上的情況
請看下面的結(jié)構(gòu):
struct MyStruct
{
double dda1;
char dda;
int type
};
對結(jié)構(gòu)MyStruct采用sizeof會出現(xiàn)什么結(jié)果呢?sizeof(MyStruct)為多少呢?也許你會這樣求:
sizeof(MyStruct)=sizeof(double)+sizeof(char)+sizeof(int)=13
但是當(dāng)在VC中測試上面結(jié)構(gòu)的大小時,你會發(fā)現(xiàn)sizeof(MyStruct)為16。你知道為什么在VC中會得出這樣一個結(jié)果嗎?
其實(shí),這是VC對變量存儲的一個特殊處理。為了提高CPU的存儲速度,VC對一些變量的起始地址做了“對齊”處理。在默認(rèn)情況下,VC規(guī)定各成員變量存放的起始地址相對于結(jié)構(gòu)的起始地址的偏移量必須為該變量的類型所占用的字節(jié)數(shù)的倍數(shù)。下面列出常用類型的對齊方式(vc6.0,32位系統(tǒng))。
各成員變量在存放的時候根據(jù)在結(jié)構(gòu)中出現(xiàn)的順序依次申請空間,同時按照上面的對齊方式調(diào)整位置,空缺的字節(jié)VC會自動填充。同時VC為了確保結(jié)構(gòu)的大小為結(jié)構(gòu)的字節(jié)邊界數(shù)(即該結(jié)構(gòu)中占用最大空間的類型所占用的字節(jié)數(shù))的倍數(shù),所以在為最后一個成員變量申請空間后,還會根據(jù)需要自動填充空缺的字節(jié)。
下面用前面的例子來說明VC到底怎么樣來存放結(jié)構(gòu)的。
struct MyStruct
{
double dda1;
char dda;
int type
};
sizeof(MyStruct)=8+1+3+4=16,
其中有3個字節(jié)是VC自動填充的,沒有放任何有意義的東西。
附:更改C編譯器的缺省字節(jié)對齊方式
在缺省情況下,C編譯器為每一個變量或是數(shù)據(jù)單元按其自然對界條件分配空間。一般地,可以通過下面的方法來改變?nèi)笔〉膶鐥l件:
· 使用偽指令#pragma pack (n),C編譯器將按照n個字節(jié)對齊。
· 使用偽指令#pragma pack (),取消自定義字節(jié)對齊方式,恢復(fù)缺省對齊。
另外,還有如下的一種方式:
· __attribute((aligned (n))),讓所作用的結(jié)構(gòu)成員對齊在n字節(jié)自然邊界上。如果結(jié)構(gòu)中有成員的長度大于n,則按照最大成員的長度來對齊。
· __attribute__ ((packed)),取消結(jié)構(gòu)在編譯過程中的優(yōu)化對齊,按照實(shí)際占用字節(jié)數(shù)進(jìn)行對齊。
七、sizeof用法總結(jié)
在VC中,sizeof有著許多的用法,而且很容易引起一些錯誤。下面根據(jù)sizeof后面的參數(shù)對sizeof的用法做個總結(jié)。
A.參數(shù)為數(shù)據(jù)類型或者為一般變量。例如sizeof(int),sizeof(long)等等。這種情況要注意的是不同系統(tǒng)系統(tǒng)或者不同編譯器得到的結(jié)果可能是不同的。例如int類型在16位系統(tǒng)中占2個字節(jié),在32位系統(tǒng)中占4個字節(jié)。
B.參數(shù)為數(shù)組或指針。下面舉例說明.
int a[50]; //sizeof(a)=4*50=200; 求數(shù)組所占的空間大小
int *a=new int[50]; // sizeof(a)=4; a為一個指針,sizeof(a)是求指針的大小,在32位系統(tǒng)中,當(dāng)然是占4個字節(jié)。
C.參數(shù)為結(jié)構(gòu)或類。sizeof應(yīng)用在類和結(jié)構(gòu)的處理情況是相同的。但有兩點(diǎn)需要注:
第一、結(jié)構(gòu)或者類中的靜態(tài)成員不對結(jié)構(gòu)或者類的大小產(chǎn)生影響,因?yàn)殪o態(tài)變量的存儲位置與結(jié)構(gòu)或者類的實(shí)例地址無關(guān)。
第二、沒有成員變量的結(jié)構(gòu)或類的大小為1,因?yàn)楸仨毐WC結(jié)構(gòu)或類的每一個實(shí)例在內(nèi)存中都有唯一的地址。
下面舉例說明:
Class Test{int a;static double c}; //sizeof(Test)=4.
Test *s; //sizeof(s)=4,s為一個指針。
Class test1{ }; //sizeof(test1)=1;
D. 參數(shù)為其他。下面舉例說明。
int func(char s[5]);
{
cout << s ; //函數(shù)的參數(shù)在傳遞的時候系統(tǒng)處理為一個指針,所以sizeof(s)實(shí)際上為求指針的大小。
return 1;
}
sizeof(func( "1234" ))=4; //因?yàn)閒unc的返回類型為int,所以相當(dāng)于求sizeof(int).
一、sizeof的概念
sizeof是C語言的一種單目操作符,如C語言的其他操作符++、--等。它并不是函數(shù)。sizeof操作符以字節(jié)形式給出了其操作數(shù)的存儲大小。
操作數(shù)可以是一個表達(dá)式或括在括號內(nèi)的類型名。操作數(shù)的存儲大小由操作數(shù)的類型決定。
二、sizeof的使用方法
1、用于數(shù)據(jù)類型
sizeof使用形式:sizeof(type)
數(shù)據(jù)類型必須用括號括住。如sizeof(int),還有sizeof int ,或sizeof(variable)三種方式?!?br>
2、用于變量
sizeof使用形式:sizeof(var_name)或sizeof var_name
變量名可以不用括號括住。如sizeof (var_name),sizeof var_name等都是正確形式。帶括號的用法更普遍,大多數(shù)程序員采用這種形式?!?BR> 注意:sizeof操作符不能用于函數(shù)類型,不完全類型或位字段。不完全類型指具有未知存儲大小的數(shù)據(jù)類型,如未知存儲大小的數(shù)組類型、未知內(nèi)容的結(jié)構(gòu)或聯(lián)合類型、void類型等?!?BR> 如sizeof(max)若此時變量max定義為int max(),sizeof(char_v) 若此時char_v定義為char char_v [MAX]且MAX未知,sizeof(void)都不是正確形式。
三、sizeof的結(jié)果
sizeof操作符的結(jié)果類型是size_t,它在頭文件中typedef為unsigned int類型。該類型保證能容納實(shí)現(xiàn)所建立的最大對象的字節(jié)大小?!?BR> 1、若操作數(shù)具有類型char、unsigned char或signed char,其結(jié)果等于1?!?BR> ANSI C正式規(guī)定字符類型為1字節(jié)。
2、int、unsigned int 、short int、unsigned short 、long int 、unsigned long 、float、double、long double類型的sizeof
在ANSI C中沒有具體規(guī)定,大小依賴于實(shí)現(xiàn),一般可能分別為2、2、2、2、4、4、4、8、10?!?BR> 3、當(dāng)操作數(shù)是指針時,sizeof依賴于編譯器。例如Microsoft C/C++7.0中,near類指針字節(jié)數(shù)為2,far、huge類指針字節(jié)數(shù)為4。一般Unix的指針字節(jié)數(shù)為4。
4、當(dāng)操作數(shù)具有數(shù)組類型時,其結(jié)果是數(shù)組的總字節(jié)數(shù)。
5、聯(lián)合類型操作數(shù)的sizeof是其最大字節(jié)成員的字節(jié)數(shù)。結(jié)構(gòu)類型操作數(shù)的sizeof是這種類型對象的總字節(jié)數(shù),包括任何墊補(bǔ)在內(nèi)。
讓我們看如下結(jié)構(gòu):
struct A{
char b;
double x;
} a;
在某些機(jī)器上sizeof(a)=12,而一般sizeof(char)+ sizeof(double)=9?!?BR> 這是因?yàn)榫幾g器在考慮對齊問題時,在結(jié)構(gòu)中插入空位以控制各成員對象的地址對齊。如double類型的結(jié)構(gòu)成員x要放在被4整除的地址?!?BR> 6、如果操作數(shù)是函數(shù)中的數(shù)組形參或函數(shù)類型的形參,sizeof給出其指針的大小?!?br>
四、sizeof與其他操作符的關(guān)系
sizeof的優(yōu)先級(各操作符的優(yōu)先級可參考C語言運(yùn)算符優(yōu)先級列表(超詳細(xì)):)為2級,比/、%等3級運(yùn)算符優(yōu)先級高。它可以與其他操作符一起組成表達(dá)式。如i*sizeof(int);其中i為int類型變量?!?br>
五、sizeof的主要用途
1、sizeof操作符的一個主要用途是與存儲分配和I/O系統(tǒng)那樣的例程進(jìn)行通信。例如:
void *malloc(size_t size),
size_t fread(void *ptr, size_t size, size_t nmemb,FILE * stream)?!?BR> 2、sizeof的另一個的主要用途是計(jì)算數(shù)組中元素的個數(shù)。例如:
void * memset(void *s,int c,sizeof(s))?!?br>
六、建議
由于操作數(shù)的字節(jié)數(shù)在實(shí)現(xiàn)時可能出現(xiàn)變化,建議在涉及到操作數(shù)字節(jié)大小時用sizeof來代替常量計(jì)算。
本文主要包括二個部分,第一部分重點(diǎn)介紹在VC中,怎么樣采用sizeof來求結(jié)構(gòu)的大小,以及容易出現(xiàn)的問題,并給出解決問題的方法,
第二部分總結(jié)出VC中sizeof的主要用法。
1、 sizeof應(yīng)用在結(jié)構(gòu)上的情況
請看下面的結(jié)構(gòu):
struct MyStruct
{
double dda1;
char dda;
int type
};
對結(jié)構(gòu)MyStruct采用sizeof會出現(xiàn)什么結(jié)果呢?sizeof(MyStruct)為多少呢?也許你會這樣求:
sizeof(MyStruct)=sizeof(double)+sizeof(char)+sizeof(int)=13
但是當(dāng)在VC中測試上面結(jié)構(gòu)的大小時,你會發(fā)現(xiàn)sizeof(MyStruct)為16。你知道為什么在VC中會得出這樣一個結(jié)果嗎?
其實(shí),這是VC對變量存儲的一個特殊處理。為了提高CPU的存儲速度,VC對一些變量的起始地址做了“對齊”處理。在默認(rèn)情況下,VC規(guī)定各成員變量存放的起始地址相對于結(jié)構(gòu)的起始地址的偏移量必須為該變量的類型所占用的字節(jié)數(shù)的倍數(shù)。下面列出常用類型的對齊方式(vc6.0,32位系統(tǒng))。
各成員變量在存放的時候根據(jù)在結(jié)構(gòu)中出現(xiàn)的順序依次申請空間,同時按照上面的對齊方式調(diào)整位置,空缺的字節(jié)VC會自動填充。同時VC為了確保結(jié)構(gòu)的大小為結(jié)構(gòu)的字節(jié)邊界數(shù)(即該結(jié)構(gòu)中占用最大空間的類型所占用的字節(jié)數(shù))的倍數(shù),所以在為最后一個成員變量申請空間后,還會根據(jù)需要自動填充空缺的字節(jié)。
下面用前面的例子來說明VC到底怎么樣來存放結(jié)構(gòu)的。
struct MyStruct
{
double dda1;
char dda;
int type
};
sizeof(MyStruct)=8+1+3+4=16,
其中有3個字節(jié)是VC自動填充的,沒有放任何有意義的東西。
附:更改C編譯器的缺省字節(jié)對齊方式
在缺省情況下,C編譯器為每一個變量或是數(shù)據(jù)單元按其自然對界條件分配空間。一般地,可以通過下面的方法來改變?nèi)笔〉膶鐥l件:
· 使用偽指令#pragma pack (n),C編譯器將按照n個字節(jié)對齊。
· 使用偽指令#pragma pack (),取消自定義字節(jié)對齊方式,恢復(fù)缺省對齊。
另外,還有如下的一種方式:
· __attribute((aligned (n))),讓所作用的結(jié)構(gòu)成員對齊在n字節(jié)自然邊界上。如果結(jié)構(gòu)中有成員的長度大于n,則按照最大成員的長度來對齊。
· __attribute__ ((packed)),取消結(jié)構(gòu)在編譯過程中的優(yōu)化對齊,按照實(shí)際占用字節(jié)數(shù)進(jìn)行對齊。
七、sizeof用法總結(jié)
在VC中,sizeof有著許多的用法,而且很容易引起一些錯誤。下面根據(jù)sizeof后面的參數(shù)對sizeof的用法做個總結(jié)。
A.參數(shù)為數(shù)據(jù)類型或者為一般變量。例如sizeof(int),sizeof(long)等等。這種情況要注意的是不同系統(tǒng)系統(tǒng)或者不同編譯器得到的結(jié)果可能是不同的。例如int類型在16位系統(tǒng)中占2個字節(jié),在32位系統(tǒng)中占4個字節(jié)。
B.參數(shù)為數(shù)組或指針。下面舉例說明.
int a[50]; //sizeof(a)=4*50=200; 求數(shù)組所占的空間大小
int *a=new int[50]; // sizeof(a)=4; a為一個指針,sizeof(a)是求指針的大小,在32位系統(tǒng)中,當(dāng)然是占4個字節(jié)。
C.參數(shù)為結(jié)構(gòu)或類。sizeof應(yīng)用在類和結(jié)構(gòu)的處理情況是相同的。但有兩點(diǎn)需要注:
第一、結(jié)構(gòu)或者類中的靜態(tài)成員不對結(jié)構(gòu)或者類的大小產(chǎn)生影響,因?yàn)殪o態(tài)變量的存儲位置與結(jié)構(gòu)或者類的實(shí)例地址無關(guān)。
第二、沒有成員變量的結(jié)構(gòu)或類的大小為1,因?yàn)楸仨毐WC結(jié)構(gòu)或類的每一個實(shí)例在內(nèi)存中都有唯一的地址。
下面舉例說明:
Class Test{int a;static double c}; //sizeof(Test)=4.
Test *s; //sizeof(s)=4,s為一個指針。
Class test1{ }; //sizeof(test1)=1;
D. 參數(shù)為其他。下面舉例說明。
int func(char s[5]);
{
cout << s ; //函數(shù)的參數(shù)在傳遞的時候系統(tǒng)處理為一個指針,所以sizeof(s)實(shí)際上為求指針的大小。
return 1;
}
sizeof(func( "1234" ))=4; //因?yàn)閒unc的返回類型為int,所以相當(dāng)于求sizeof(int).
相關(guān)文章
C++ 中實(shí)現(xiàn)把EXCEL的數(shù)據(jù)導(dǎo)入數(shù)據(jù)庫(ACCESS、MSSQL等)實(shí)例代碼
這篇文章主要介紹了C++ 中實(shí)現(xiàn)把EXCEL的數(shù)據(jù)導(dǎo)入數(shù)據(jù)庫(ACCESS、MSSQL等)實(shí)例代碼的相關(guān)資料,需要的朋友可以參考下2017-04-04C語言實(shí)現(xiàn)linux網(wǎng)卡檢測改進(jìn)版
這篇文章主要為大家詳細(xì)介紹了C語言實(shí)現(xiàn)linux網(wǎng)卡檢測的改進(jìn)版,具有一定的參考價值,感興趣的小伙伴們可以參考一下2018-06-06C語言實(shí)現(xiàn)打印數(shù)組以及打印注意事項(xiàng)說明
這篇文章主要介紹了C語言實(shí)現(xiàn)打印數(shù)組以及打印注意事項(xiàng)說明,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-01-01