3. 字符数组练习
sizeof 只关注占用内存空间的大小,单位是字节,不关心内存中存放的是什么
sizeof 是操作符
strlen是求字符串长度的,统计的是\0之前出现的字符个数,一定要找到\0才算结束,所以可能存在越界访问的
strlen是库函数
3.1 对于数组结尾没有‘\0’的情况, sizeof 和strlen
3.1.1 sizeof
代码验证
#include<stdio.h>
int main()
{
//字符数组
char arr[] = { 'a','b','c','d','e','f' }; //该数组没有'\0'
printf("%d\n", sizeof(arr)); //arr单独放入sizeof中,表示整个数组的大小 6个元素,单个字符1字节,所以是6字节。
printf("%d\n", sizeof(arr + 0)); //arr没有单独放入,表示首元素地址+0还它本身,首元素地址大小 4字节/32位机器 或 8字节/64位机器
printf("%d\n", sizeof(*arr));//对首元素地址解引用,得到第一个元素‘a',大小1字节,
printf("%d\n", sizeof(arr[1]));//arr[1]表示第二个元素,大小1字节
printf("%d\n", sizeof(&arr));//数组名取地址,表示整个数组的地址,是地址大小 就是4字节/32位机器 或 8字节/64位机器
printf("%d\n", sizeof(&arr + 1));//数组名取地址,表示整个数组,&arr+1跳过整个数组,指向整个数组之后的位置,指针类型是char(*)[6] ,地址的大小就是 4字节/32位机器 或 8字节/64位机器
printf("%d\n", sizeof(&arr[0] + 1)); //arr[0]是首元素对首元素取地址,得到的是第一个元素地址,然后&arr[0]+1 表示的是第二个元素地址,大小是4字节/32位机器 或 8字节/64位机器
return 0;
}
代码分析
sizeof(arr)
sizeof(arr + 0);
sizeof(*arr));
sizeof(arr[1])
sizeof(&arr)
sizeof(&arr + 1)
sizeof(&arr[0] + 1)
3.1.2 strlen
首先需要明白,strlen求字符串长度,是数字符串结尾’\0’前面有几个字符
代码验证
#include<stdio.h>
int main()
{
//字符数组
char arr[] = { 'a','b','c','d','e','f' }; //该数组结尾没有'\0'
printf("%d\n", strlen(arr)); //求数组的长度,由于数组结尾没有'\0'所以算出来的是随机值,
printf("%d\n", strlen(arr + 0));//和上面一条一样,随机值
printf("%d\n", strlen(*arr));//对首元素解引用 是'a’strlen('a')->strlen(97),非法访问-err
printf("%d\n", strlen(arr[1]));//和上一个一样 'b'-98,和上面的代码类似,是非法访问 - err
printf("%d\n", strlen(&arr));//对整个数组取地址,得到的虽然是地址,但是仍然没有'\0',所以还是随机值
printf("%d\n", strlen(&arr + 1));//对整个数组取地址+1 跳过整个数组,但是也是从头开始算长度,仍然没有'\0',所以还是随机值
printf("%d\n", strlen(&arr[0] + 1));//&arr[0]得到 第一个元素a 再+1 得到b 求长度,也是随机值,因为没有'\0'
return 0;
}
代码分析
strlen(arr)
strlen(arr + 0)
strlen(*arr)//error
strlen(arr[1])//error
strlen(&arr)
strlen(&arr + 1)
strlen(&arr[0] + 1)
3.2 对于数组结尾有‘\0’的情况, sizeof 和strlen
#include<stdio.h>
#include<string.h>
int main()
{
char arr[] = "abcdef"; //数组是7个元素,后面还有一个'\0'
printf("%d\n", sizeof(arr)); //arr单独在sizeof中表示求整个数组的大小 :7字节
printf("%d\n", sizeof(arr + 0)); //arr不是单独放在sizeof中,表示首元素地址+0还是首元素地址,大小:4字节/32位机器 或 8字节/64位机器
printf("%d\n", sizeof(*arr));//arr不是单独放在sizeof中,表示首元素地址,对首元素地址解引用,得到第一个元素‘a’ 大小:1字节
printf("%d\n", sizeof(arr[1]));//arr[1]表示第二个元素’b’ 大小:1字节
printf("%d\n", sizeof(&arr));//&arr表示整个数组的地址,是地址大小就是: 4字节/32位机器 或 8字节/64位机器
printf("%d\n", sizeof(&arr + 1));// &arr表示整个数组的地址 +1就是跳过整个数组,指向数组之后的位置,该指针类型是char(*)[6] ,地址大小就是: 4字节/32位机器 或 8字节/64位机器
printf("%d\n", sizeof(&arr[0] + 1));//&arr[0]表示第一个元素'a' + 1 表示第二个元素的地址,大小:4字节/32位机器 或 8字节/64位机器
printf("%d\n", strlen(arr));//求字符串长度,长度:6字节
printf("%d\n", strlen(arr + 0));//和上一条一样,长度:6字节
//printf("%d\n", strlen(*arr));//对arr解引用,得到'a',是97,传给strlen是一个非法的地址,造成非法访问
//printf("%d\n", strlen(arr[1]));//arr[1]是第二个元素,‘b'是98, 传给strlen是一个非法的地址,造成非法访问
printf("%d\n", strlen(&arr));//&arr表示对整个数组取地址,得到整个数组地址(值是首地址的值,意义是整个数组)求该地址长度,大小是:6字节
printf("%d\n", strlen(&arr + 1));//&arr表示对整个数组取地址,+1 得到跳过整个数组的地址,求该地址长度,是随机值,因为不清楚'\0'在哪里
printf("%d\n", strlen(&arr[0] + 1));//&arr[0]表示第一个元素'a'+1得到’b' 从第二个元素往后统计字符串长度,大小:5字节
return 0;
}
3.2.1 sizeof 代码分析
sizeof(arr)
sizeof(arr + 0)
sizeof(*arr)
sizeof(arr[1])
sizeof(&arr)
sizeof(&arr + 1)
sizeof(&arr[0] + 1)
3.2.2 strlen 代码分析
strlen(arr)
strlen(arr + 0)
strlen(*arr)//ERROR
strlen(arr[1]//error
strlen(&arr)
strlen(&arr + 1)
strlen(&arr[0] + 1)