一.sizeof
sizeof是一个关键字,是编译时运行符(值在编译时就计算好了),用于判断变量或者数据类型的字节大小.
sizeof(data type);
其中,data type是要计算大小的数据类,参数可以是数组,指针,类型,对象,函数等
由于在编译时计算,因此sizeof不能用来返回动态分配的内存空间的大小。实际上,用sizeof来返回类型以及静态分配的对象、结构或数组所占的空间,返回值跟对象、结构、数组所存储的内容没有关系。
具体而言,当参数分别如下时,sizeof返回的值表示的含义如下:
- 数组:编译时分配的数组空间大小
- 指针:存储该指针所用的空间大小(存储该指针的地址的长度,是长整型,应该为4)
- 类型:该类型所占的空间大小
- 对象:对象的实际占用空间大小
- 函数:函数的返回类型所占的空间大小.函数的返回类型不能是void
二.strlen
strlen是函数,要在运行时才能计算,参数必须是字符型指针.计算字符串的长度,直到遇到'\0'字符才结束,但不包括'\0';
size_t strlen(const char* str);
函数参数:地址(需要计算长度的字符串)
三.sizeof与strlen的区别
- sizeof是关键字,strlen是函数
- sizeof的参数可以是数组,指针,类型,对象,函数等.strlen的参数只能是字符指针,且必须以'\0'结尾
- 数组名做sizeof的参数不退化,做strlen的参数会退化为指针
- 编译器在编译时就计算出了sizeof的结果,而strlen函数必须在运行时才能计算出来.并且sizeof计算的是数据类型占内存的大小,而strlen计算的是字符串实际的长度.
四.有关常见题解析
int main() {
char str[] = { 'c','s','w','t','y','\0' };
printf("%d\n", sizeof(str)); //6:包括;'\0'
printf("%d\n", strlen(str)); //5:不包括'\0'
return 0;
}
int main() {
int a[] = { 1,2,3,4 };
printf("%d\n", sizeof(a)); //16:计算数组大小
printf("%d\n", sizeof(a + 0)); //4:计算的是数组首元素的大小
printf("%d\n", sizeof(*a)); //4:数组名表示数组首元素地址,因此计算的也是数组首元素大小
printf("%d\n", sizeof(a + 1)); //4:数组名表示数组首元素的地址(指针),指针+1同样是指针,因此计算的是一个指针变量的大小
printf("%d\n", sizeof(a[1])); //4:计算的是数组第二个元素的大小
printf("%d\n", sizeof(&a)); //4:&a表示的是数组的地址(数组指针),因此计算的是一个指针变量的大小
printf("%d\n", sizeof(*&a)); //16:数组指针解引用,计算的是数组的大小
printf("%d\n", sizeof(&a + 1)); //4:数组指+1表示的同样是指针,因此计算的也就是一个指针变量的大小
printf("%d\n", sizeof(&a[0])); //4:数组首元素的地址(指针)
printf("%d\n", sizeof(&a[0] + 1)); //4:数组首元素的地址(指针)+1
char arr[] = { 'a','b','c','d','e','f' };
printf("%d\n", sizeof(arr)); //6
printf("%d\n", sizeof(arr + 0));//1
printf("%d\n", sizeof(*arr));//1
printf("%d\n",sizeof(arr[1])); //1
printf("%d\n", sizeof(&arr)); //4
printf("%d\n", sizeof(&arr + 1));//4
printf("%d\n", sizeof(&arr[0] + 1));//4
printf("%d\n", strlen(arr)); //未知的,因为strlen遇到'\0'才会结束.该字符数组中没有'\0',因此结果是未知的
printf("%d\n", strlen(arr + 0)); //同理
//printf("%d\n", strlen(*arr)); //strlen的参数只能是指针类型
//printf("%d\n", strlen(arr[1]));
printf("%d\n", strlen(&arr)); //同理
printf("%d\n", strlen(&arr + 1)); //同理
printf("%d\n", strlen(&arr[0] + 1)); //同理
char arr[] = "abcdef";
printf("%d\n", sizeof(arr)); //7
printf("%d\n", sizeof(arr + 0)); //4
printf("%d\n", sizeof(*arr)); //1
printf("%d\n", sizeof(arr[1]));//1
printf("%d\n", sizeof(&arr));//4
printf("%d\n", sizeof(&arr + 1));//4
printf("%d\n", sizeof(&arr[0] + 1));//4
printf("%d\n", strlen(arr));//6
printf("%d\n", strlen(arr + 0));//6
printf("%d\n", strlen(*arr));//出现错误
printf("%d\n", strlen(arr[1])); //出现错误
printf("%d\n", strlen(&arr)); //6
printf("%d\n", strlen(&arr + 1)); //未知结果
printf("%d\n", strlen(&arr[0] + 1));//5
char *p = "abcdef";
printf("%d\n", sizeof(p)); //4
printf("%d\n", sizeof(p + 1)); //4
printf("%d\n", sizeof(*p)); //1
printf("%d\n", sizeof(p[0]));//1
printf("%d\n", sizeof(&p));//4
printf("%d\n", sizeof(&p + 1));//4
printf("%d\n", sizeof(&p[0] + 1));//4
printf("%d\n", strlen(p)); //6
printf("%d\n", strlen(p + 1)); //5
printf("%d\n", strlen(*p)); //出现错误
printf("%d\n", strlen(p[0])); //出现错误
printf("%d\n", strlen(&p)); //对指针变量取地址,得到新的地址,因此结果是未知的
printf("%d\n", strlen(&p + 1)); //未知的地址+1,因此同样是未知的
printf("%d\n", strlen(&p[0] + 1)); //5
return 0;
}