数组的总结
假设全部题是在32位操作系统下。
例1: int a[] = {1,2,3,4};
printf("%d\n",sizeof(a)); //16
printf("%d\n",sizeof(a+0)); //4 数组名隐式转成指针,指针加一个数得到还是int* ,32位系统中指针占4个字节。
printf("%d\n",sizeof(*a)); //4 数组名已经隐式转成指针int*了,再解引用就变成了int,int占4个字节。
printf("%d\n",sizeof(a+1)); //4 同第二行。
printf("%d\n",sizeof(a[1])); //4 去下标得到int。
printf("%d\n",sizeof(&a)); //4 数组指针,只要是指针,在32位系统下就是4字节
printf("%d\n",sizeof(*&a)); //16 &a是数组指针,对其解引用是从指向的位置往后找int(*)[4]类型对应的字节数。
printf("%d\n",sizeof(&a+1)); //4 数组指针加一还是数组指针,指针就是4字节。
printf("%d\n",sizeof(&a[0])); //4 数组取下标,再取地址得到int*。
printf("%d\n",sizeof(&a[0]+1)); //4 同上
例2:字符数组 char arr[] = {‘a’,‘b’,‘c’,‘d’,‘e’,‘f’};
sizeof:
printf("%d\n", sizeof(arr)); //6 sizeof是求字符数组长度,在数组中定义几个就是几。
printf("%d\n", sizeof(arr+0)); //4 数组名隐式转成指针char*,指针就是4个字节。
printf("%d\n", sizeof(*arr)); //1 得到一个char。
printf("%d\n", sizeof(arr[1])); //1 得到一个char。
printf("%d\n", sizeof(&arr)); //4 数组指针char*,还是指针4个字节。
printf("%d\n", sizeof(&arr+1)); //4 数组指针加一还是char*,还是4字节。
printf("%d\n", sizeof(&arr[0]+1)); //4 &arr[0]得到的是char*,char*加一还是char*,指针4字节。
strlen
printf("%d\n", strlen(arr)); //随机数 未定义行为,因为strlen是找'\0',字符数组中没有‘\0',会在内存中一直数,直到找到'\0'。
printf("%d\n", strlen(arr+0)); //同上
printf("%d\n", strlen(*arr)); //未定义行为,strlen函数的参数为char*,而这里*arr是char。
printf("%d\n", strlen(arr[1])); //同上
printf("%d\n", strlen(&arr)); //未定义行为 同第一行
printf("%d\n", strlen(&arr+1)); //未定义行为 同上
printf("%d\n", strlen(&arr[0]+1)); //未定义行为。
例3:char arr[] = “abcdef”;
sizeof:
printf("%d\n", sizeof(arr)); //7 sizeof求所占内存的大小,这次包含'\0'。
printf("%d\n", sizeof(arr+0)); //4 数组名隐式转成指针char*。
printf("%d\n", sizeof(*arr)); //1 *arr得到char
printf("%d\n", sizeof(arr[1])); //1 同上
printf("%d\n", sizeof(&arr)); //4 数组指针,指针就是4字节。
printf("%d\n", sizeof(&arr+1)); //4 还是数组指针。
printf("%d\n", sizeof(&arr[0]+1)); //4 &arr[0]得到char*,还是指针,指针加一还是指针。
strlen:
printf("%d\n", strlen(arr)); //6 '\0'只是一个截止符,不算个数。
printf("%d\n", strlen(arr+0)); //6 同上
printf("%d\n", strlen(*arr)); //未定义行为 strlen的参数是char*,这里传个了char,编译错误。
printf("%d\n", strlen(arr[1])); //同上。
printf("%d\n", strlen(&arr)); //6 碰巧&arr跟arr代表的地址一样,都是从'a'开始找'\0',结果也就都是6。
printf("%d\n", strlen(&arr+1)); //未定义行为 跳过刚刚那个数组,下段地址中不一定有'\0'。
printf("%d\n", strlen(&arr[0]+1)); //5 从‘b’开始找'\0'。
例4:char *p = “abcdef”;
sizeof:
printf("%d\n", sizeof(p)); //4 p是char*
printf("%d\n", sizeof(p+1)); //4 指针加一还是指针
printf("%d\n", sizeof(*p)); //1 *p得到char
printf("%d\n", sizeof(p[0])); //1 同上
printf("%d\n", sizeof(&p)); //4 p是char*,&p得到char**,是一个二级指针,还是指针
printf("%d\n", sizeof(&p+1)); //4 指针加一还是指针
printf("%d\n", sizeof(&p[0]+1)); //4 p[0]得到char,&p[0]得到char*,char*加一还是指针。
strlen:
printf("%d\n", strlen(p)); //6 p指向一个字符串,strlen找'\0'
printf("%d\n", strlen(p+1)); //5 跳过了第一个字符。
printf("%d\n", strlen(*p)); //未定义行为 strlen的参数是char*,这里却是char。
printf("%d\n", strlen(p[0])); //同上
printf("%d\n", strlen(&p)); //随机数 未定义行为,从
printf("%d\n", strlen(&p+1)); //随机数 未定义行为
printf("%d\n", strlen(&p[0]+1)); //5 跳过了第一个字符
例5:二维数组 int a[3][4] = {0};
printf("%d\n",sizeof(a)); //48 3*4*4
printf("%d\n",sizeof(a[0][0])); //4 得到int
printf("%d\n",sizeof(a[0])); //16 得到一个类型为int[4]的元素,4*4
printf("%d\n",sizeof(a[0]+1)); //4 a[0]得到int[4],再加一隐式转成指针int*,a[0]+1相当于a[0][1]。
printf("%d\n",sizeof(*(a[0]+1))); //4 根据上一行得到int*,再解引用就是int。
printf("%d\n",sizeof(a+1)); //4 指针加一还是指针
printf("%d\n",sizeof(*(a+1))); //16 同倒数第二行
printf("%d\n",sizeof(&a[0]+1)); //4 a[0]得到int[4],&a[0]得到int(*)[4],再加一还是数组指针。
printf("%d\n",sizeof(*(&a[0]+1))); //16 跟上一行同样,解引用后就得到了a[1][0]~a[1][3]这一行元素。
printf("%d\n",sizeof(*a)); //16 *a相当于*(a+0),相当于a[0],同第三行一样。
printf("%d\n",sizeof(a[100])); //16 sizeof在编译阶段就算出结果了,所以不存在越界错误,越界是在运行阶段的错误