您的位置:首页 > 其它

strlen和sizeof的区别

2013-04-09 20:52 246 查看
首先应该明确的概念是sizeof是操作符,而不是函数。strlen是一个函数。

sizeof操作符返回的是参数所占的内存数,而strlen函数返回的是参数的字符数长度,不是所占用的内存的大小。需要注意的是strlen函数的参数是字符串,并且必须以串结束符“\0”结尾。

比如说:char str[8] = "fdsa";

sizeof(str) = 8字节 //字符数组str所占用的内存大小

strlen(str) = 4 //初始化的字符长度

再比如:char *str = "Hello";

strlen(str) = 5, //初始化的字符长度

sizeof(str) = 4; //str是一个指针,指针在32位的计算机中占用4字节,在64位计算机中占用8字节

sizeof(*str) = 1; //*str表示字符串首地址的内容,在此就是字符H,占用内存大小为1字节

再看一个例子: char str[] = "0123456789";

sizeof(str) = 11; //str是数组,计算到了"\0"的位置,以此是(10+1)

strlen(str) = 10; //strlen不计算"\0"。

例4: int ss[100] = "0123456789";

sizeof(ss) = 400; //表示ss内存大小,4*100

strlen(ss) 错误,因为strlen的参数只能是char *,且必须是以"\0"结尾。

补充1:

void Func(char a[100])
{
 cout<< sizeof(a) << endl; // 4字节而不是100字节
}
在参数的调用过程中,数组退化为指针。所以这里是4字节。

补充2:

int main()
{

int a[100];
int *p = NULL;
printf("%d %d %d %d %d %d\n",sizeof(p),sizeof(*p),sizeof(a),sizeof(a[100]),sizeof(&a),sizeof(&a[0]));

return 0;
}
运行结果:



要注意sizeof(a[100])等于4,因为a[100]只是数组中的一个元素。

补充3:

下面就考虑一下这个问题:

int main()
{
char a[1000];
int i;
for(i=0; i<1000; i++)
{
a[i] = -1-i;
}
printf("%d",strlen(a));
return 0;
}

运行结果如下:



此题看上去真的很简单,但是却鲜有人答对。答案是 255。别惊讶,我们先分析分析。for 循环内,当 i 的值为 0 时,a[0]的值为-1。关键就是-1 在内存里面如何存储。我们知道在计算机系统中,数值一律用补码来表示(存储)。主要原因是使用补码,可以将符号位和其它位统一处理;同时,减法也可按加法来处理。另外,两个用补码表示的数相加时,如果最高位(符号位)有进位,则进位被舍弃。正数的补码与其原码一致;负数的补码:符号位为 1,其余位为该数绝对值的原码按位取反,然后整个数加 1。按照负数补码的规则,可以知道-1的补码为
0xff,-2 的补码为 0xfe……当 i 的值为 127时,a[127]的值为-128,而-128 是 char 类型数据能表示的最小的负数。当 i 继续增加,a[128]的值肯定不能是-129。因为这时候发生了溢出,-129 需要 9 位才能存储下来,而 char 类型数据只有 8 位,所以最高位被丢弃。剩下的 8 位是原来 9 位补码的低 8 位的值,即 0x7f。当 i 继续增加到 255 的时候,-256 的补码的低 8 位为 0。然后当 i 增加到 256 时,-257 的补码的低 8 位全为
1,即低八位的补码为 0xff,如此又开始一轮新的循环……按照上面的分析,a[0]到 a[254]里面的值都不为 0,而 a[255]的值为 0。strlen 函数是计算字符串长度的,并不包含字符串最后的‘\0’。而判断一个字符串是否结束的标志就是看是否遇到‘\0’。如果遇到‘\0’,则认为本字符串结束。分析到这里,strlen(a)的值为 255 应该完全能理解了。这个问题的关键就是要明白 char类型默认情况下是有符号的,其表示的值的范围为[-128,127],超出这个范围的值会产生溢出。另外还要清楚的就是负数的补码怎么表示。弄明白了这两点,这个问题其实就很简单了。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: