在平时的C语言学习中,有很多朋友可能对sizeof()和strlen()的用法混淆不清,今天lucky同学以一维数组的访问为例,来对两者的用法加以介绍和区分。
sizeof()的用法
sizeof不是函数,是一个操作符,sizeof计算的是对象所占内存的大小,单位是字节,它不在乎内存中存放的是什么,只在乎内存的大小。
对于数组的访问,首先要清楚这几个概念:
数组名通常为首元素的地址,但有两个例外:
1、sizeof(数组名),这里的数组名表示整个数组,计算的是整个数组的大小
2、&数组名,这里的数组名表示整个数组,取出的是整个数组的地址(数组的地址也是地址,在32位平台下占4个字节,在64位平台下占8个字节)
搞清楚上面的概念后,来看如下几个例子:
#include <stdio.h>
int main()
{
int a[] = {1, 2, 3, 4};
printf("%d\n",sizeof(a)); // 16
//sizeof(a),此时的数组名表示整个数组的大小,四个整型即16字节
printf("%d\n",sizeof(a+0)); // 4/8
//sizeof里放的不是单独的a,则a代表首元素的地址,(a+0)也是数组首元素的地址,所以为4或8字节
printf("%d\n",sizeof(*a)); // 4
//a表示数组首元素的地址,*a表示数组的第一个元素,sizeof(*a)就是第一个元素的大小
printf("%d\n",sizeof(a+1)); // 4/8
//a表示数组首元素的地址,a+1是第二个元素的地址,sizeof(a+1)就是第二个元素的地址的大小
printf("%d\n",sizeof(a[1])); // 4
//表示数组的第二个元素的大小
printf("%d\n",sizeof(&a)); // 4/8
//&a取出的是首元素地址的地址,也是地址,大小就是4或8个字节
printf("%d\n",sizeof(*&a)); // 16
//&a取得是整个数组大小,再*就拿到整个数组,计算的是整个数组的大小
printf("%d\n",sizeof(&a+1)); // 4/8
&a拿到的是整个数组的地址,加1跳过整个数组,产生的是4后面位置的地址,还是一个地址
printf("%d\n",sizeof(&a[0])); // 4/8
//&a[0]表示数组第一个元素的地址
printf("%d\n",sizeof(&a[0]+1)); // 4/8
//&a[0]+1表示数组第二个元素的地址
return 0;
}
以上是整形数组,再来看字符数组:
#include <stdio.h>
int main()
{
char arr[] = {'a','b','c','d','e','f'}; // [a b c d e f]
printf("%d\n", sizeof(arr)); // 6
//sizeof(arr),这里的数组名表示整个字符数组,一个字符为1个字节
printf("%d\n", sizeof(arr+0)); // 4/8
//arr+0 是数组首元素的地址
printf("%d\n", sizeof(*arr)); // 1
//*arr指的是数组的首元素'a',为一个字节
printf("%d\n", sizeof(arr[1])); // 1
//arr[1]表示数组第二个元素'b',为一个字节
printf("%d\n", sizeof(&arr)); // 4/8
//&arr取得是整个数组的地址,是地址,还是占4或8个字节
printf("%d\n", sizeof(&arr+1)); // 4/8
//(&arr + 1) 是从数组的地址开始向后跨过了整个数组,而产生的一个地址,是地址,还是占4或8个字节
printf("%d\n", sizeof(&arr[0]+1)); // 4/8
//(&arr[0]+1) 表示数组第二个元素的地址
return 0;
}
strlen()的用法
strlen () 是一个库函数,主要是用来求字符串长度(内存里面只能放字符),从给定的地址一直向后访问字符,直到遇到 \0 为止,strlen() 统计的是 \0 之前字符的个数。strlen () 函数的定义: size_t strlen(const char* string);可以看到 strlen() 函数里面放的是一个const char*类型的指针变量,所以 () 里面默认是一个地址。
了解清楚 strlen() 的概念,再来看以下例子来深入理解:
#include <stdio.h>
int main()
{
char arr[] = {'a','b','c','d','e','f'}; // [a b c d e f]
printf("%d\n", strlen(arr)); // 随机值
//arr表首元素地址,即从首元素开始数,arr数组中没有\0,所以strlen函数会继续往后找\0,统计\0之前出现的字符个数,由于不知道何时才会找到\0,所以为随机值
printf("%d\n", strlen(arr+0)); // 随机值
//和上面一样,(arr+0) 还是第一个元素的地址,仍然是从第一个元素开始数,strlen函数会继续往后找\0,统计\0之前出现的字符个数
printf("%d\n", strlen(*arr)); //err 非法访问
//这里指的是第一个元素字符a,ASCII为97,strlen函数从97的地址开始数,访问了错误的地址,出现了野指针,属于非法访问
printf("%d\n", strlen(arr[1])); //err 非法访问
//同上面一样,这里指的是第二个元素字符b,仍然属于非法访问
printf("%d\n", strlen(&arr)); // 随机值
//&arr取得是整个数组的地址,仍然是从数组首元素地址开始,向后找遇到\0之前字符的个数,所以为随机值
printf("%d\n", strlen(&arr+1)); // 随机值
//(&arr+1)是从数组地址开始向后跳过了整个数组而产生的一个地址,从这个地址开始,向后找遇到\0之前字符的个数,所以仍是随机值
printf("%d\n", strlen(&arr[0]+1)); // 随机值
//(&arr[0]+1) 指的是数组第二个元素的地址,从这个地址向后找\0之前字符的个数,不难知道仍是随机值
return 0;
}
到这里,大家对两者的用法有了更加深入的了解了吗,get it !