C语言--指针和数组试题

一、 数组名的意义:
1. sizeof(数组名),这里的数组名表示整个数组,计算的是整个数组的大小。
2. &数组名,这里的数组名表示整个数组,取出的是整个数组的地址。
3. 除此之外所有的数组名都表示首元素的地址。

//一维数组
int main()
{
	//一维数组
	int a[] = { 1, 2, 3, 4 };       //32平台地址大小是4,64平台地址大小是8.
	printf("%d\n", sizeof(a));      //16,sizeof(数组名)--数组名表示整个数组
	printf("%d\n", sizeof(a + 0));  //4,只有两种情况数组名表示整个数组,其他情况数组名都是首元素地址。4/8,
	                                       //1.sizeof(数组名)--数组名表示整个数组.
	                                       //2.&数组名--数组名表示整个数组
	printf("%d\n", sizeof(*a));     //4,首元素地址解应用,代表是首元素。
	printf("%d\n", sizeof(a + 1));  //4,表示第二个元素地址4/8,
	printf("%d\n", sizeof(a[1]));   //4,表示第二个元素大小
	printf("%d\n", sizeof(&a));     //4,取出的是数组的地址,地址的大小为4/8,
	printf("%d\n", sizeof(*&a));    //16,计算的是整个数组的大小
	printf("%d\n", sizeof(&a + 1)); //4,取出的数组的地址+1,跳过一个数组地址大小,所以还是数组的大小4/8,
	printf("%d\n", sizeof(&a[0]));  //4,取出第一个元素的地址,4/8
	printf("%d\n", sizeof(&a[0] + 1));//4,取出第二个元素的地址,4/8

	//字符数组
	char arr[] = { 'a', 'b', 'c', 'd', 'e', 'f' };
	printf("%d\n", sizeof(arr));         //6,计算的是数组的大小,数组大小为6*1
	printf("%d\n", sizeof(arr + 0));     //4,计算的是第一个元素地址的大小4/8
	printf("%d\n", sizeof(*arr));        //1,计算的是第一个元素的大小
	printf("%d\n", sizeof(arr[1]));      //1, 计算的是第二个元素地址
	printf("%d\n", sizeof(&arr));        //4, 取出的是数组的地址,地址的大小为4/8,
	printf("%d\n", sizeof(&arr + 1));    //4,取出的数组的地址+1,跳过一个数组地址大小,所以还是数组的大小4/8,
	printf("%d\n", sizeof(&arr[0] + 1)); //4,计算第二个元素地址
	//strlen函数
	printf("%d\n", strlen(arr));        //随机值,求字符串长度函数,是寻找\0之前的值,但字符数组无\0
	printf("%d\n", strlen(arr + 0));    //随机值,
	printf("%d\n", strlen(*arr));       //程序崩溃,*arr是字符a,strlen函数需要传入地址,a的地址为97,输入非法访问,程序崩溃
	printf("%d\n", strlen(arr[1]));     //程序崩溃
	printf("%d\n", strlen(&arr));       //随机值,求得是数组的地址,但是还是找不到\0
	printf("%d\n", strlen(&arr + 1));   //随机值,找不到\0
	printf("%d\n", strlen(&arr[0] + 1));//随机值,找不到\0


	//字符串数组
	char arr[] = "abcdef";           //常量字符串存入数组是[a b c d e f \0]形式。
	printf("%d\n", sizeof(arr));     //7,还计算的\0的大小,因此是七个字符
	printf("%d\n", sizeof(arr + 0)); //4,计算地址的大小,4/8
	printf("%d\n", sizeof(*arr));    //1,计算第一个元素的大小
	printf("%d\n", sizeof(arr[1]));  //1,计算的是第二个元素的大小
	printf("%d\n", sizeof(&arr));    //4,计算地址的大小,4/8
	printf("%d\n", sizeof(&arr + 1));//4,计算地址的大小,4/8
	printf("%d\n", sizeof(&arr[0] + 1));//4,计算地址的大小,4/8
	//strlen函数
	printf("%d\n", strlen(arr));     //6
	printf("%d\n", strlen(arr + 0)); //6
	printf("%d\n", strlen(*arr));    //错误,strlen接收的是地址,不是具体的元素
	printf("%d\n", strlen(arr[1]));  //错误,strlen接收的是地址,不是具体的元素
	printf("%d\n", strlen(&arr));    //6
	printf("%d\n", strlen(&arr + 1));//随机值
	printf("%d\n", strlen(&arr[0] + 1));//5


	//字符串指针
	char *p = "abcdef";             //是把常量字符串abcdef的a的地址放到指针p里面去
	printf("%d\n", sizeof(p));      //4,计算指针变量p的大小
	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,取出指针变量p的地址
	printf("%d\n", sizeof(&p + 1)); //4,取出指针变量p的地址
	printf("%d\n", sizeof(&p[0] + 1));//4
	//strlen函数
	printf("%d\n", strlen(p));    //6
	printf("%d\n", strlen(p + 1));//5
	printf("%d\n", strlen(*p));   //错误,strlen接收的是地址,不是具体的元素
	printf("%d\n", strlen(p[0])); //错误,strlen接收的是地址,不是具体的元素
	printf("%d\n", strlen(&p));   //随机值
	printf("%d\n", strlen(&p + 1));//随机值
	printf("%d\n", strlen(&p[0] + 1));//5


	//二维数组
	int a[3][4] = { 0 };
	printf("%d\n", sizeof(a));            //48,计算整个数组的大小
	printf("%d\n", sizeof(a[0][0]));      //4,计算的是第一个元素的大小
	printf("%d\n", sizeof(a[0]));         //16,计算的是第一行的元素,a[0]单独放到sizeof中是第一行的数组名
	printf("%d\n", sizeof(a[0] + 1));     //4,计算的是第一行第二个元素的地址
	printf("%d\n", sizeof(*(a[0] + 1)));  //4,计算的是第一行第二个元素大小
	printf("%d\n", sizeof(a + 1));        //4,计算的是第二行地址
	printf("%d\n", sizeof(*(a + 1)));     //16,计算的是第二行数组中元素大小
	printf("%d\n", sizeof(&a[0] + 1));    //4,计算的是第二行地址,a[0]是第一行的地址,&a[0]就是第一行地址
	printf("%d\n", sizeof(*(&a[0] + 1))); //16,计算的是第二行元素大小
	printf("%d\n", sizeof(*a));           //16,计算的是第一行元素大小
	printf("%d\n", sizeof(a[3]));         //16,计算的是第四行地址
}

二、指针试题详解

指针代码1
int main()
{
	int a[5] = { 1, 2, 3, 4, 5 };
	int *ptr = (int *)(&a + 1);
	//&a是数组指针类型,+1跳过整个数组,通过(int *)类型强制转换,存放到整数类型指针int *ptr中
	printf("%d,%d", *(a + 1), *(ptr - 1));
	// *(a + 1)=2,得到的是第二个元素
	// *(ptr - 1)=5,整形指针类型指针-1,向后移动一个整形
	return 0;
}


指针代码2
//已知,结构体Test类型的变量大小是20个字节
//假设p的值为0x100000。 如下表表达式的值分别为多少?
struct Test
{
	int Num;
	char *pcName;
	short sDate;
	char cha[2];
	short sBa[4];
}*p;
int main()
{
	p = (struct Test *) 0x100000;
	printf("%p\n", p + 0x1);//结构体指针+1,增加20个字节,0x100000+0x14=0x100014
	printf("%p\n", (unsigned long)p + 0x1);//无符号整形+1,1048576+1=1048577=0x100001
	printf("%p\n", (unsigned int*)p + 0x1);//无符号整形指针+1,0x100000+0x4=0x100004
	//根据指针类型判断+1,跳过多少字节
	return 0;
}


指针代码3
int main()
{
	int a[4] = { 1, 2, 3, 4 };
	int *ptr1 = (int *)(&a + 1);
	int *ptr2 = (int *)((int)a + 1);//地址的存储:01 00 00 00   02 00 00 00  03 00 00 00   04 00 00 00小端存储
	printf("%x,%x", ptr1[-1], *ptr2);
	// ptr1[-1]==ptr1+(-1),ptr1[-1]=4
	//*ptr2=20 00 00 00
	return 0;
}


指针代码4
int main()
{
	int a[3][2] = { (0, 1), (2, 3), (4, 5) };//逗号表达式计算的是后一个结果{1,3,5}
	int *p;
	p = a[0];//存放的是第一行的地址
	printf("%d", p[0]);//输出的是第一行第一个值,p[0]=1
	return 0;
}


指针代码5
int main()
{
	int a[5][5];
	int(*p)[4];//数组指针指向的是类型是四个元素
	p = a;//
	printf("%p,%d\n", &p[4][2] - &a[4][2], &p[4][2] - &a[4][2]);//FFFFFFFC,-4
	//p[4][2]=*(*(p+4)+2)
	//%p,-4原码1000 0000 0000 0000 0000 0000 0000 0100
	//     反码1111 1111 1111 1111 1111 1111 1111 1011
	//     补码1111 1111 1111 1111 1111 1111 1111 1100
	return 0;
}


指针代码6
int main()
{
	int aa[2][5] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };
	int *ptr1 = (int *)(&aa + 1);
	int *ptr2 = (int *)(*(aa + 1));
	printf("%d,%d", *(ptr1 - 1), *(ptr2 - 1));
	//*(ptr1 - 1)=10, *(ptr2 - 1)=5
	return 0;
}


指针代码7
int main()
{
	char *a[] = { "work", "at", "alibaba" };//都是把每个数字串的首地址存到字符指针数组中
	char**pa = a;
	pa++;
	printf("%s\n", *pa);
	//*pa=at
	return 0;
}


指针代码8
int main()
{
	char *c[] = { "ENTER", "NEW", "POINT", "FIRST" };
	char**cp[] = { c + 3, c + 2, c + 1, c };
	char***cpp = cp;
	printf("%s\n", **++cpp);//POINT
	printf("%s\n", *--*++cpp + 3);//ER
	printf("%s\n", *cpp[-2] + 3);//ST
	printf("%s\n", cpp[-1][-1] + 1); //EW
	return 0;
}

代码8结构图 

 

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值