数组指针和函数指针的难点解析

**1、int (*parr3 [10] ) [5]
这个首先为一个数组,(parr3 [10] )可看作(a),则整体为数组指针数组
2、数组元素打印的程序

#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
void print(int *p, int sz)
{
	for (int i = 0; i < sz; i++)
	{
		printf("%d\n", *(p + i));
	}
}

int main( )
{
	int arr[10] = { 1, 2, 3, 4, 5, 6, 7, 8, 9 };
	int *p = arr;
	int sz = sizeof(arr) / sizeof(arr[0]);
	//一级指针p,传给函数
	print(p, sz);
	system("pause");
	return 0;
}

3、void (pfun1) ( ); 函数指针
void pfun2 ( ); 函数声明,返回值void
(void (
() ( ))0)( ); 对0强转为函数指针,整个语句为函数的调用
void (signal (int ,void ()(int) ))(int ) (int ,void (
)(int)为函数的参数列表(int 和函数指针), signal为函数名,void(
) (int)为其返回值,返回
*值为函数指针。上式亦可定义为
typedef void (*pfun_t) (int) 这个为函数指针的重定义
pfun_t signal ( int ,pfun_t)
4、 函数指针数组:把函数的地址存在一个数组中,那这个数组就叫函数指针数组。
int (parr1[10])( ); 首先parr1为一个数组,里面存放的int ()()类型是的函数指针

**5、计算器的加减乘除可用函数指针数组来表示,比较简单,定义一个函数指针数组:int (p[5])(int x,int y)={0,add,sub,mul,div};(x,y为输入的两个数,p[5]为数组名,调用的时候为ret=(p[input]) (x,y)😉
6、回调函数:回调函数就是一个通过函数指针调用的函数。

7、指针笔试题:
笔试题1:

int main()
{
int a[5] = { 1, 2, 3, 4, 5 };
int *ptr = (int *)(&a + 1);
printf( "%d,%d", *(a + 1), *(ptr - 1));
return 0;
}

解析:2和5
在这里插入图片描述
a表示数组首元素的地址,+1则移动到2对应的位置,则*(a+1)输出结果为2,&a为整个数组的地址,&a+1则加到数组的末尾,将这个地址强转为int 并赋值给ptr,则(ptr-1)的结果为5.

笔试题2

struct Test
{
int Num;
char *pcName;
short sDate;
char cha[2];
short sBa[4];
}*p;
//假设p 的值为0x100000。 如下表表达式的值分别为多少?
int main()
{
printf("%p\n", p + 0x1);
printf("%p\n", (unsigned long)p +0x1);
printf("%p\n", (unsigned int*)p + 0x1);
return 0;
}

//由于还没学习结构体,这里告知结构体的大小是20个字节

解析:结果为0x100014, 0x100001,0x100004
p为一个结构体指针,故p+0x1表示的为加整个结构体的大小,则结果为0x100014;(加了20,16进制则为0x14). (unsigned long)p +0x1把p强转为整形,则结果为0x100001;(unsigned int*)p + 0x1则为首地址+1,则结果为0x100004.

笔试题3

int main()
{
int a[4] = { 1, 2, 3, 4 };
int *ptr1 = (int *)(&a + 1);
int *ptr2 = (int *)((int)a + 1);
printf( "%x,%x", ptr1[-1], *ptr2);
return 0;
}

解析:结果为4和2000000
在这里插入图片描述

笔试题4

#include <stdio.h>
int main(int argc, char * argv[])
{
int a[3][2] = { (0, 1), (2, 3), (4, 5) };
int *p;
p = a[0];
printf( "%d", p[0]);
}

**解析:结果为1,**int a[3][2] = { (0, 1), (2, 3), (4, 5) };为逗号语句,结果以最终的为主。
在这里插入图片描述

笔试题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]);
return 0;
}

解析:结果为fffffffc,-4
在这里插入图片描述

笔试题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));
return 0;

解析:结果为10和5
在这里插入图片描述
笔试题7

#include <stdio.h>
int main()
{
char *a[] = {"work","at","alibaba"};
char**pa = a;
pa++;
printf("%s\n", *pa);
return 0;
}

解析:结果为at
在这里插入图片描述

笔试题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);
printf("%s\n", *--*++cpp+3);
printf("%s\n", *cpp[-2]+3);
printf("%s\n", cpp[-1][-1]+1);
return 0;
}

解析:结果为:POINT,ER,ST,EW

在这里插入图片描述

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值