今天下雨了,来更一下指针的进阶知识。
今天主要内容就是:
1. 字符指针
2. 数组指针
3. 指针数组
4. 数组传参和指针传参
5. 函数指针
6. 函数指针数组
一. 字符指针
1.字符指针char*,来看一段代码

第一行代码的本质其实是把字符串 hello bit. 首字符的地址放到了pstr中。

来看这样一段代码

这里str3和str4指向的是一个同一个常量字符串。C/C++会把常量字符串存储到单独的一个内存区域,当几个指针指向同一个字符串的时候,他们实际会指向同一块内存。但是用相同的常量字符串去初始化相同的数组的时候就会开辟出不同的内存块。所以str1和str2不同,str3和str4相同。
二.指针数组
1.指针数组本质上就是存放指针的数组,它是数组!!!

三.数组指针
1.数组指针是指针!!!它是能够指向数组的指针!!!
对于第二行代码:p先和*结合,说明p是一个指针变量,然后指着指向的是一个大小为10个整型的数组。所以p是一个 指针,指向一个数组,叫数组指针。
2.&数组名VS数组名
&arr 表示的是数组的地址,而不是数组首元素的地址,
而arr代码数组首元素地址(两个例外)前面这种&arr是第一种,sizeof(arr)是第二种。
3.数组指针的使用

分辨一下下面的代码

四.数组参数,指针参数
1.一维数组传参,如下面代码

这里需要注意的就是test2(int **arr)是否可以传参,因为arr2是指针数组,内含20个int*类型的指针,对于指针传参,使用二级指针。
2.二维数组传参,依旧看代码


红框内其实不常见 ,这样写非常别扭。
3.一级指针传参
 
 
4.二级指针传参

五.函数指针
1.函数指针是指向函数的指针

两个红框内的都是函数的地址,虽然写法不同,但意义完全一样。(相交于数组),接下来如果要把地址存起来就这样写:
 
 
那么函数指针该怎么用呢?这里一定要注意:要对*p加括号!
这个红框内的执行语句与int ret=Add(2,3);其实是等价的,所以调用函数指针的时候还可以这么写:

 
这是函数指针类型。
看两端代码的意思
1.(*(void (*)())0)();


2.void (   *signal (int , void(*)(int) )   ) ( int );
这个代码写的太麻烦了,可以做typedef

红框内是语法要求,不能写成 typedef void(*)(int) pf_t 。
六.函数指针数组
1.注意如下代码,函数指针数组的写法和调用的写法

接下来看看函数指针数组的具体应用,图1和图2代码相比,图1使用了函数指针数组,使得代码简单。
int Add(int x, int y)
{
	return x + y;
}
int Sub(int x, int y)
{
	return x - y;
}
int Mul(int x, int y)
{
	return x * y;
}
int Div(int x, int y)
{
	return x / y;
}
void menu()
{
	printf("**************************\n");
	printf("****  1.add   2.sub   ****\n");
	printf("****  3.mul   4.div   ****\n");
	printf("****  0.exit          ****\n");
	printf("**************************\n");
}
int main()
{
	int input = 0;
	int x = 0;
	int y = 0;
	int ret = 0;
	//转移表
	int (*pfArr[])(int, int) = { 0, Add, Sub, Mul, Div };
	do
	{
		menu();
		printf("请选择:>");
		scanf("%d", &input);
		if (input == 0)
		{
			printf("退出计算器\n");
		}
		else if (input >= 1 && input <= 4)
		{
			printf("请输入2个操作数:>");
			scanf("%d%d", &x, &y);
			ret = pfArr[input](x, y);
			printf("ret = %d\n", ret);
		}
		else
		{
			printf("选择错误\n");
		}
	} while (input);
	return 0;
}
int Add(int x, int y)
{
	return x + y;
}
int Sub(int x, int y)
{
	return x - y;
}
int Mul(int x, int y)
{
	return x * y;
}
int Div(int x, int y)
{
	return x / y;
}
void menu()
{
	printf("**************************\n");
	printf("****  1.add   2.sub   ****\n");
	printf("****  3.mul   4.div   ****\n");
	printf("****  0.exit          ****\n");
	printf("**************************\n");
}
int main()
{
	int input = 0;
	int x = 0;
	int y = 0;
	int ret = 0;
	do
	{
		menu();
		printf("请选择:>");
		scanf("%d", &input);
		switch (input)
		{
		case 1:
			printf("请输入2个操作数:>");
			scanf("%d%d", &x, &y);
			ret = Add(x, y);
			printf("ret = %d\n", ret);
			break;
		case 2:
			printf("请输入2个操作数:>");
			scanf("%d%d", &x, &y);
			ret = Sub(x, y);
			printf("ret = %d\n", ret);
			break;
		case 3:
			printf("请输入2个操作数:>");
			scanf("%d%d", &x, &y);
			ret = Mul(x, y);
			printf("ret = %d\n", ret);
			break;
		case 4:
			printf("请输入2个操作数:>");
			scanf("%d%d", &x, &y);
			ret = Div(x, y);
			printf("ret = %d\n", ret);
			break;
		case 0:
			printf("退出计算器\n");
			break;
		default:
			printf("选择错误\n");
			break;
		}
	} while (input);
	return 0;
}










