&取自小破站鹏哥指针笔试面试题p42
#include <stdio.h>
#include <string.h>
int main()
{
int arr[] = { 1,2,3,4 };
printf("%d\n", sizeof(arr));//16 int字节*4==16
printf("%d\n", sizeof(arr + 0));//8 结果:32位:4 64位:8(本机为64位置)
printf("%d\n", sizeof(*arr));//4 arr首元素地址,*arr首元素
printf("%d\n", sizeof(arr + 1));//8 第二个元素地址
printf("%d\n", sizeof(arr[1]));//4 第二个元素大小
printf("%d\n", sizeof(&arr));//8 !!!难 4/8,取出数组的地址,地址的大小为4/8
printf("%d\n", sizeof(*&arr));//16 数组的地址解引用,就是整个数组(取地址和解引用抵消)
printf("%d\n", sizeof(&arr + 1));//8 取地址&arr后地址加1,跳过整个地址,还是一个地址,地址大小为4/8
printf("%d\n", sizeof(&arr[0]));//8 4/8,首元素地址
printf("%d\n", sizeof(&arr[0] + 1));//8 第二个元素的地址
printf("\n");
char a[] = { 'a','b','c','d','e','f' };
printf("%d\n", sizeof(a));//6 char一字节,6*1
printf("%d\n", sizeof(a + 0));//8 首元素地址+0,还是首元素的地址大小4/8
printf("%d\n", sizeof(*a));//1 a首元素地址,*a首元素
printf("%d\n", sizeof(a[1]));//1 第二个元素大小
printf("%d\n", sizeof(&a));//8 难!!! &arr数组地址,地址大小就为4/8
printf("%d\n", sizeof(&a + 1));//8 +1再跨越整个地址大小,还是一个地址大小4/8
printf("%d\n", sizeof(&a[0] + 1));//8 第二个元素地址
printf("\n");
char b[] = { 'a','b','c','d','e','f' };
printf("%d\n", strlen(b));//随机值x 从首地址开始找,找到/0才停止
printf("%d\n", strlen(b + 0));//随机值x 首元素地址+0,和上者一样
//printf("%d\n", strlen(*b));
// //error b:首元素地址,*b:首元素为a,a的ASCLL码值为97,原式=strlen(97):读取位置0x00000061时发生访问冲突
//printf("%d\n", strlen(b[1]));
//error 相当于strlen(98)
printf("%d\n", strlen(&b));//随机值x
printf("%d\n", strlen(&b + 1));//随机值x-6 但与上者相差6
printf("%d\n", strlen(&b[0] + 1));//随机值x-1 与上上者相差1
printf("\n");
char c[] = "abcdef";
printf("%d\n", sizeof(c));//7 实际存放a b c d e f \0 所占空间大小7*1
printf("%d\n", sizeof(c+0));//8 首元素地址大小4/8
printf("%d\n", sizeof(*c));//1 首元素大小
printf("%d\n", sizeof(c[1]));//1 第二个元素大小
printf("%d\n", sizeof(&c));//8 虽然是数组地址,但也是地址,所以4/8
printf("%d\n", sizeof(&c+1));//8 跳过整个地址,但地址大小不变,4/8
printf("%d\n", sizeof(&c[0]+1));//8 第二个元素的地址
printf("\n");
char d[] = "abcdef";
printf("%d\n", strlen(d));//6
printf("%d\n", strlen(d+0));//6
//printf("%d\n", strlen(*d));//error strlen(97):读取位置0x00000061时发生访问冲突
//printf("%d\n", strlen(d[1]));//erroe strlen(98)
printf("%d\n", strlen(&d));//6 警告,但能运行
printf("%d\n", strlen(&d+1));//随机值 跳过一整个字符数组后,无法预测何时有\0
printf("%d\n", strlen(&d[0]+1));//5 第二个元素b后,\0之前
printf("\n");
char* e = "abcdef";//仅把a的地址放在e中
printf("%d\n", sizeof(e));//8 计算指针变量的大小4/8
printf("%d\n", sizeof(e+1));//8 字符b的地址
printf("%d\n", sizeof(*e));//1 对指针e进行解引用,得到字符a
printf("%d\n", sizeof(e[0]));//1 !!! e[0]==*(e+0)=='a'
printf("%d\n", sizeof(&e));//8 取地址4/8
printf("%d\n", sizeof(&e+1));//8 跳过一个字符数组的地址,新地址4/8
printf("%d\n", sizeof(&e[0]+1));//8 b的地址
printf("\n");
char* f = "abcdef";
printf("%d\n", strlen(f));//6 首字符之后,\0之前
printf("%d\n", strlen(f+1));//5 第二个字符之后,\0之前
//printf("%d\n", strlen(*f));//error f为指针,*f就是a,原式=strlen(a)报错
//printf("%d\n", strlen(f[0]));//error *f与f[0]同理
printf("%d\n", strlen(&f));//随机值!!!f指针指向数组首地址后,&:又对f的地址求长度,直到\0
printf("%d\n", strlen(&f+1));//随机值 f地址后跳一个数组长度,直到\0
printf("%d\n", strlen(&f[0]+1));//5 首元素后一个地址
printf("\n");
int g[3][4] = { 0 };
//***二位数组的首元素地址就是第一行的地址
printf("%d\n", sizeof(g));//48 3*4*4
printf("%d\n", sizeof(g[0][0]));//4
printf("%d\n", sizeof(g[0]));//16 g[0]代表第一行,第一行有4个元素,4*4(调用二维数组不能省略列)
printf("%d\n", sizeof(g[0]+1));//8 g[0]指向第一行首元素地址,再加1,就是第一行第二个元素地址4/8
printf("%d\n", sizeof(*(g[0] + 1)));//4 第一行第二个元素,大小为4字节
printf("%d\n", sizeof(g+1));//8 g是首元素地址,二维数组的首元素是其第一行,则g就是g[0],g+1就是第二行地址
printf("%d\n", sizeof(*(g + 1)));//16 相当于sizeof(g[1])
printf("%d\n", sizeof(&g[0]+1));//8 第一行地址+1为第二行地址
printf("%d\n", sizeof(*(&g[0] + 1)));//16 第二行的字节大小
printf("%d\n", sizeof(*g));//16 g是首元素地址,*g就是第一行,再求字节大小
printf("%d\n", sizeof(g[3]));//16 ***sizeof不参与实际计算(不对数组进行访问),只看数组类型,所以g[3]和g[0]无异
return 0;
}
总结:
//int array[2][3] = { {1, 2, 3},{4, 5, 6} };二维数组的初始化,中间需要花括号,而不是括号!!!
// strlen求解字符串的长度,任何一个字符串的结尾都会隐藏一个\0,他是字符串的结束标志但\0不算做一个字符
// sizeof是一个运算符,主要用来计算所占空间字节的大小。包括计算\0
//sizeof(数组名):表示整个数组
//&(数组名):表示整个数组
//其他均表示首元素地址
//无论什么类型int,char。。。地址大小都是4/8