C語言中sizeof 與strlen 區別
一.本質區別
sizeof 和strlen 有本質上的區別。sizeof 是C 語言的一種單目運算符,如++、--等,並不是函數,sizeof 的優先級為2 級,比/、% 等3 級運算符優先級高,sizeof
以字節的形式給出操作數的存儲空間的大小。而 strlen 是一個函數,是由 C 語言
的標准庫提供的。strlen 計算的 是字符串的長度。
二.使用區別
1.sizeof
sizeof 的操作數可以是數據類型、函數、變量,表達式使用方式為:
(1)數據類型
sizeof (type )
例如我們要計算一個int 型數據的存儲空間可以用: sizeof (int)。需要注意的是
sizeof 的操作數是數據類型時要加括號。其數值大小為該數據類型所占的存儲空
間的字節數。
(2)變量
sizeof(變量名)
如果定義 int a ,可以使用 sizeof (a )計算a 變量占據的存儲空間。具體大小
與a 的類型有關。
注意:由於sizeof 是操作符sizeof a或sizeof (a )都可以。(可以不使用括號),
如果操作數是數組名則給出數組所占用內存的字節數。如果數組名做函數的參數
傳遞時退化為指針。
(3)表達式
sizeof (表達式)
sizeof 可以對一個表達式求值,編譯器根據表達式的最終結果類型來確定大小,
一般不會對表達式進行計算。例如:sizeof(1+1.5)
(4)函數調用
sizeof(函數名())
sizeof 也可以對一個函數調用求值,其結果是函數返回類型的大小,函數並不會
被調用,舉例來說定義如下函數:
int myprint ()
{
printf(“hello\n” );
return 0;
}
int main()
{
printf(“%d”,sizeof(mypaint()));
return 0;
}
結果只打印函數返回類型的sizeof 值,並沒有打印hello說明函數myprint並沒有
調用。
C99 標准規定,函數、不能確定類型的表達式以及位域(bit-field)成員不能被
計算sizeof 值,即下面這些寫法都是錯誤的:
如:sizeof(myprint)(注意 sizeof(myprint()是可以的))
或者sizeof 一個void 返回類型的函數如:
void foo () { }
sizeof( foo () );
以及位域:
struct S
{
unsigned int f1 : 1;
unsigned int f2 : 5;
unsigned int f3 : 12;
};
sizeof( S.f1 );
2.strlen
strlen 的應用則不像 sizeof 那么廣泛,strlen 的參數必須是char * 的指針,如果用
strlen 計算數據類型 strlen(int)這種用法是錯誤的。strlen 的計算必須依賴字符
序列中的’\0’ 字符,strlen 就是通過判斷是否遇到’\0’ 來判斷字符序列是否結束
的。
它的計算原理類似於下面的兩條語句
while(*p!=’\0’)
length++
strlen 的用法:分為以下幾種參數
(1)char * 指針
strlen(指針名)
如果參數是指針則計算該指針指向字符序列的長度。(以’\0’ 作為判斷標志)例如:
定義char *p=“hello world”;strlen(p)=11,而 sizeof (p)=4。可以看到 strlen
計算的是指針指向的字符串的長度而sizeof 計算的是指針本身所占用的內存空
間的大小。
(2)數組
strlen(數組名)
如果參數是數組的話,實際傳遞的是一個指針,strlen 會按照上面處理指針的模
式處理該數組。
我們可以看下面的例子:
char a[]=”hh”;
strlen(a);
很顯然strlen 的結果是2。但是如果數組是這樣賦值的呢?
char a[]={‘h’,’h’};
strlen(a);
那么現在strlen(a )的結果又是多少呢?這個數就不一定了,原因是 strlen 會去
計算a 地址開始的字符串的長度,由於前一種賦值方式會將hh以字 符串的形式
賦值給數組會將字符串結束符’\0’ 一同賦值,這時strlen 就會檢查到結束符停止
計算,而第二種復值方式是以單個字符的形式賦值沒有結束 符’\0’, 這時我們用
sizeof 得到的結果是正常的,而用 strlen 由於找不到結束符,會繼續的計算直到
找到結束符為止。所以這個數是不確定.
-------------------------
One example
#define PATH_TMP "12345"
static void test_sizeof_strlen() {
char *msg = "12345";
printf("sizeof(msg)---%d\n" , sizeof(msg));
printf("strlen(msg)---%d\n" , strlen(msg));
char array[] = "12345";
printf("sizeof(array)---%d\n" , sizeof(array));
printf("strlen(array)---%d\n" , strlen(array));
printf("sizeof(PATH_TMP)---%d\n" , sizeof(PATH_TMP));
printf("strlen(PATH_TMP)---%d\n" , strlen(PATH_TMP));
}
The output result:
sizeof(msg)---4
strlen(msg)---5
sizeof(array)---6
strlen(array)---5
sizeof(PATH_TMP)---6
strlen(PATH_TMP)---5
------------------------------------------------------------------------------------------------------------------------
sizeof是C語言的一種單目操作符,如C語言的其他操作符++、--等。
二、sizeof的使用方法
1、用於數據類型
sizeof使用形式: sizeof(type) 數據類型必須用括號括住: sizeof(int)
2、用於變量
sizeof使用形式: sizeof(var_name) 或 sizeof var_name 變量名可以不用括號括住.如sizeof (var_name),sizeof var_name等都是正確形式
注意:sizeof操作符不能用於函數類型,不完全類型或位字段。
三、sizeof的結果(以下結果都是在Linux v2.6 gcc v4獲取)
sizeof操作符的結果類型是size_t
1、ANSI C正式規定字符類型為1字節。 sizeof(char) = 1;
4、當操作數具有數組類型時,其結果是數組的總字節數。
7、當操作數是函數中的數組形參或函數類型的形參:
四、sizeof與其他操作符的關系
sizeof的優先級為2級,比/、%等3級運算符優先級高。
1、主要用途是與存儲分配和I/O系統那樣的例程進行通信。
2、另一個的主要用途是計算數組中元素的個數。
-----------------------------------------------
sizeof和strlen的區別
例1:char ss[100]=”0123456789”;
Sizeof(ss)結果為100,ss表示在內存中預分配的大小,100*1;
Strlent(ss)結果為10,它的內部實現是用一個循環計算字符串的長度,直到”\0”為止。
例2:int ss[100]=”0123456789”;
Sizeof(ss)結果為400,ss表示在內存中的大小,100*4;
Strlen(ss)錯誤,strlen的參數只能是char*,且必須是以”\0”結尾的。
總結sizeof和strlen的區別
⒈sizeof操作符的結果類型是size_t,它在頭文件中的typedef為unsigned int 類型。該類型保證能容納實現所建立的最大對象的字節大小。
⒉sizeof是運算符,strlen是函數。
⒊sizeof可以用類型做參數,strlen只能用char*做參數,且必須是以”\0”結尾的。sizeof還可以用函數做參數,比如: short f();
printf(“%d\n”,sizeof(f()));
輸出的結果是sizeof(short),即2。
⒋數組做sizeof的參數不退化,傳遞給strlen就退化為指針。
⒌大部分編譯程序在編譯的時候就把sizeof計算過了,是類型或是變量的長度。這就是sizeof(x)可以用來定義數組位數的原因。
char str[20]=”0123456789”;
int a=strlen(satr); //a=10;
int b=sizeof(str); //b=20;
⒍strlen的結果要在運行的時候才能計算出來,用來計算字符串的長度,而不是類型占內存的大小。
⒎sizeof后如果是類型必須加括號,如果是變量名可以不加括號。這是因為sizeof是個操作符而不是函數。
⒏當使用一個結構類型或變量時,sizeof返回實際的大小。當使用一靜態的空間數組時,sizeof返回全部數組的尺寸。Sizeof操作符不能返回被動態分配的數組或外部的數組的尺寸。
⒐數組作為參數傳給函數時傳得是指針而不是數組,傳遞的是數組的首地址。在C++里傳遞數組永遠都是傳遞指向數組首元素的指針,編譯器不知道數組的大小,如果想在函數內知道數組的大小需要在函數里面用memcpy將數組復制出來,長度由另一個形參傳禁區。
⒑計算結構變量的大小就必須討論數據對其問題。
⒒sizeof操作符不能用於函數類型,不完全類型或位字段。不完全類型指具有未知存儲大小數據的數據類型,如未知存儲大小的數組類型,未知內容的結構或聯合類型,void類型等。
Sizeof的使用場合
首先要明確sizeof不是函數,也不是一元運算符,它是個類似宏定義的特殊關鍵字,sizeof()。括號內的內容在編譯過程中是不被編譯的,而是被替代類型,如int a=8; sizeof(a)。在編譯過程中,不管a的值是什么,知識被替換成類型sizeof(int),結果為4。如果sizeof(a=6)呢?也是一樣地轉換成a的類型,但是要注意,因為a=6是不被編譯的,所以執行sizeof(a=6)后,a的值還是8,是不變的。
① sizeof操作符的一個主要用途是與存儲分配和I/O系統那樣的例程進行通信。例如:
void *malloc(sizex_t size),
size_t fread(void * ptr, size_t size, size_t nmemb, FILE * stream)
② 用它可以看看某種類型的對象在內存中所占的單元。例如:
void * memset(void * s, int c, sizeof(s))
③在動態分配一對象時,可以讓系統知道要分配多少內存。
④便於一些類型的擴充。在Windows中有很多結構類型就有一個專門的字段用來存放該類型的字節大小。
⑤由於操作數的字節數在實現時可能出現變化,建議在涉及操作數字節大小時用sizeof代替常量計算。
⑥如果操作數是函數中的數組形參或函數類型的形參,sizeof給出其指針的大小。
結論:
① unsigned影響的知識最高位bit的意義(正/負),數據長度是不會被改變的,所以:
sizeof(unsigned int) == sizeof(int);
② 自定義類型的sizeof取值等同於它的類型原形。如:
typedef short WORD;
sizeof(short) == sizeof(WORD);
③ 對函數使用sizeof,在編譯階段會被函數返回值的類型取代。如:
int f1() {return 0;}
cout<<sizeof(f1())<<endl; //f1()返回值為int,因此會被認為是int。
④ 只要是指針,大小就是4。如:
cout<<sizeof(string*)<<endl; //4
⑤數組的大小是各維數的乘積×數組元素的大小。