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由于找不到结束符,会继续的计算直到找到结束符为止。所以这个数是不确定。

作者“成长之路”

sizeof和strlen有本质上的区别。sizeof是C语言的一种单目运算符,如++、–等,并不是函数,sizeof的优先级为2级,比/、%
等3级运算…

相关文章