sizeof、strlen、数组名与指针

文章详细探讨了C语言中sizeof和strlen函数与数组名的关系,通过实例展示了数组名在不同上下文中代表的意义,包括整个数组的地址、首元素的地址以及元素的大小。同时,区分了字符数组和一般数组在计算大小和长度时的差异,强调了非法访问和边界条件的重要性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

关于数组名我们可以总结以下三条规律

  1. &+数组名,取出的是整个数组的地址

  1. sizeof(数组名),计算的是整个数组的大小

  1. 其余出现的数组名都是代表首元素的地址

现在来看看sizeof、strlen与数组名能擦出怎么样的火花

//一维数组
int main()
{
    int a[] = { 1,2,3,4 };
    printf("%d \n", sizeof(a));//这里的a代表整个数组,sizeof计算整个数组的大小,大小为16
    printf("%d \n", sizeof(a + 0));//这里的sizeof中不仅仅放着a,所以这里的a代表首元素的地址,a+0代表首元素的地址,大小为4/8
    printf("%d \n", sizeof(*a));//这里的sizeof不仅仅放着a,所以这里的a代表首元素的地址,*a代表a[0],大小为4
    printf("%d \n", sizeof(a + 1));//这里的sizeof不仅仅放着a,所以这里的a代表首元素的地址,a+0代表a[1]的地址,大小为4/8
    printf("%d \n", sizeof(a[1]));//a[1]代表数组中第二个元素,大小为4
    printf("%d \n", sizeof(&a));//&a代表整个数组的地址,整个数组的地址也是地址,大小为4或者8
    printf("%d \n", sizeof(*&a));//*与&互为逆运算,所以*&a等价于a,所以这里计算整个数组的大小,大小为16
    printf("%d \n", sizeof(&a + 1));//&a代表整个数组的地址,整个数组的地址+1仍是地址,大小为4/8
    printf("%d \n", sizeof(&a[0]));//&a[0]代表着第一个元素的地址,大小为4/8
    printf("%d \n", sizeof(&a[0] + 1));//&a[0]+1代表第二个元素的地址,大小为4/8
    return 0;
}
//字符数组
int main()
{
    char arr[] = { 'a','b','c','d','e','f' };
    printf("%d \n", sizeof(arr));//arr单独放在sizeof内部,代表整个数组,sizeof计算整个数组的大小,大小为6
    printf("%d \n", sizeof(arr + 0));//arr不是单独放在sizeof内部,所以这里的arr代表首元素的地址,arr+0代表首元素的地址,大小为4/8
    printf("%d \n", sizeof(*arr));//*arr代表第一个元素,因为是char类型,大小为1
    printf("%d \n", sizeof(arr[1]));//arr[1]代表第一个元素,大小为1
    printf("%d \n", sizeof(&arr));//&arr代表整个数组的地址,但仍然是地址,大小为4/8
    printf("%d \n", sizeof(&arr + 1));//&arr代表整个数组的地址,整个数组的地址+1仍是地址,大小为4/8
    printf("%d \n", sizeof(&arr[0] + 1));//&arr[0]+1代表第二个元素的地址,大小为4/8
    return 0;
}
//字符数组
int main()
{
    char arr[] = { 'a','b','c','d','e','f' };
    printf("%d\n", strlen(arr));//arr代表首元素的地址,因为数组中没有\0,所以结果为随机值
    printf("%d\n", strlen(arr + 0));//arr+0代表首元素的地址,因为数组中没有\0,所以结果为随机值
    printf("%d\n", strlen(*arr));//*arr代表第一个元素,因为strlen接收的是地址,这里将第一个元素的值当做地址,进行了非法访问
    printf("%d\n", strlen(arr[1]));//arr[1]代表第一个元素,因为strlen接收的是地址,这里将第一个元素的值当做地址,进行了非法访问
    printf("%d\n", strlen(&arr));//&arr代表整个数组的地址,但是strlen会将该地址当做成第一个元素的地址,又因为数组中没有\0所以结果为随机值
    printf("%d\n", strlen(&arr + 1));//&arr代表整个数组的地址,&arr+1,地址跳过了整个数组,所以结果为随机值-6
    printf("%d\n", strlen(&arr[0] + 1));//&ar代表第二个元素地址,所以结果为随机值-1
}
//字符数组
int main()
{
    char arr[] = "abcdef";
    printf("%d \n", sizeof(arr));//这里的arr单独放在sizeof内部,代表整个数组,sizeof计算整个数组大小,因为"abcdef"中隐藏一个\0,所以大小为7
    printf("%d \n", sizeof(arr + 0));//arr+0代表首元素的地址,大小为4/8
    printf("%d \n", sizeof(*arr));//*arr代表第一个元素,大小为1
    printf("%d \n", sizeof(arr[1]));//arr[1]代表第二个元素,大小为1
    printf("%d \n", sizeof(&arr));//&arr代表整个数组的地址,大小为4/8
    printf("%d \n", sizeof(&arr + 1));//&arr代表整个数组的地址,整个数组的地址加1仍是地址,大小为4/8
    printf("%d \n", sizeof(&arr[0] + 1));//&arr[0]+1代表第二个元素的地址,大小为4/8
}
//字符数组
int main()
{
    char arr[] = "abcdef";//这里的字符串隐藏\0
    printf("%d \n", strlen(arr));//arr代表首元素地址,结果为6
    printf("%d \n", strlen(arr + 0));//arr+0代表首元素地址,结果为6
    //printf("%d\n", strlen(*arr));//*arr代表第一个元素,strlen接收的是地址,所以第一个元素被当做地址,进行了非法访问
    //printf("%d\n", strlen(arr[1]));//arr[1]代表第二个元素,strlen接收的是地址,所以第二个元素被当做地址,进行了非法访问
    printf("%d \n", strlen(&arr));//&arr代表整个数组的地址,但是strlen会将该地址当做成第一个元素的地址,结果为6
    printf("%d \n", strlen(&arr + 1));//&arr代表整个数组的地址,&arr+1,地址跳过了整个数组,所以\0也被跳过,结果为随机值
    printf("%d \n", strlen(&arr[0] + 1));//&arr[0]+1代表第二个元素的地址,结果为5
    return 0;
}
//字符数组
int main()
{
    char* p = "abcdef";
    printf("%d \n", sizeof(p));//这里的p代表指向首元素的指针,大小为4/8
    printf("%d \n", sizeof(p + 1));//p+1代表第二个元素的地址,大小为4/8
    printf("%d \n", sizeof(*p));//*p代表第一个元素,大小为1
    printf("%d \n", sizeof(p[0]));//p[0]代表第一个元素,大小为1
    printf("%d \n", sizeof(&p));//&p代表p的地址,大小为4/8
    printf("%d \n", sizeof(&p + 1));//&p+1,代表p的地址加1,大小为4/8
    printf("%d \n", sizeof(&p[0] + 1));//&p[0]+1代表第二个元素的地址,大小为4/8
    return 0;
}
//字符数组
int main()
{
    char* p = "abcdef";//这里的字符串隐藏\0
    printf("%d \n", strlen(p));//p代表首元素地址,所以结果为6
    printf("%d \n", strlen(p + 1));//p+1代表第二个元素地址,所以结果为5
    printf("%d \n", strlen(*p));//*p代表第一个元素,strlen接收的是地址,所以第一个元素被当做地址,进行了非法访问
    printf("%d \n", strlen(p[0]));//p[0]代表第一个元素,strlen接收的是地址,所以第一个元素被当做地址,进行了非法访问
    printf("%d \n", strlen(&p));//&p代表p的地址,因为不知道\0在哪,结果为随机值
    printf("%d \n", strlen(&p + 1));//&p+1代表p的地址+1,因为不知道\0在哪,结果为随机值,且跟上一个的随机值没关系
    printf("%d \n", strlen(&p[0] + 1));//&p[0]+1代表第二个元素的地址,所以结果为5
    return 0;
}
//二维数组
int main()
{
    int a[3][4] = { 0 };
    printf("%d 48\n", sizeof(a));//a单独放在sizeof内部,代表整个数组,sizeof计算整个数组的大小,大小为48
    printf("%d 4\n", sizeof(a[0][0]));//a[0][0]代表第一行第一个元素,大小为4
    printf("%d 16\n", sizeof(a[0]));//a[0]代表第一行数组的数组名,数组名单独放在sizeof内部,所以sizeof计算第一行的数组大小大小为16
    printf("%d 4\n", sizeof(a[0] + 1));//a[0]并不单独放在sizeof内部,所以这里a[0]+1代表第一行第二个元素的地址,大小为4/8
    printf("%d 4\n", sizeof(*(a[0] + 1)));//a[0]+1,解引用后代表第一行第二个元素的地址,大小为4
    printf("%d 4\n", sizeof(a + 1));//a并不单独放在sizeof内部,所以这里a+1代表第二行数组的地址,大小为4/8
    printf("%d 16\n", sizeof(*(a + 1)));//a+1代表第二行数组的地址,解引用后等价于a[1],所以大小为16
    printf("%d 4\n", sizeof(&a[0] + 1));//&a[0]+1代表第二行数组的地址,大小为4/8
    printf("%d 16\n", sizeof(*(&a[0] + 1)));//*(&a[0] + 1)等价于a[1],大小为16
    printf("%d 16\n", sizeof(*a));//*a等价于a[1],大小为16
    printf("%d 16\n", sizeof(a[3]));//a[3]代表第四行数组的大小,大小为16,这里没有造成越界访问,因为sizeof不会进行实质操作
    return 0;
}
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值