【C语言】—— 指针与数组的理解和相关应用

一、数组指针

1.理解数组指针指的是一个指向数组的指针

int main()
{
	int*  p1[10];//p先与[10]结合说明p是一个数组,再与*结合说明p是一个存了10个整形指针的数组
	int(*p2)[10];//p先与*结合,说明p是一个指针变量,
	//然后与[10]结合说明p是一个指向大小为10个整形变量的数组的数组指针
	int arr[10];
	int *arr2[10];
	int(*p)[10] = &arr;
	//p = &arr2;//不能成功,因为&arr2的类型为int*(*)[10]
	int* (*p2)[10] = &arr2;//可以成功
	return 0;
}
int main()
{
	int arr[10] = { 0 };
	printf("%p\n", arr);//输出002EF9B8
	printf("%p\n", &arr);//输出002EF9B8
	return 0;
}
int main()
{
	int arr[10] = { 0 };
	printf("%p\n", arr);//输出003EFAA8
	printf("%p\n", &arr);//输出003EFAA8

	printf("%p\n", arr + 1);//003EFAAC加一跳过4个字节,即一个元素
	printf("%p\n", &arr + 1);//003EFAD0加一跳过40个字节,即整个数组
	return 0;
}

注:上述代码说明虽然arr与&arr值是一样的,但是所表示的意思是不一样的
实际上&arr指的是数组的地址,arr指的是数组首元素的地址,因此arr+1跳过一个元素的大小,&arr+1跳过一个数组的大小

数组指针的使用

void print_arr1(int (*p)[5], int row, int col)
{
	int i = 0;
	for (i = 0; i < row; i++)
	{
		int j = 0;
		for (j = 0; j < col; j++)
		{
			printf("%d ", p[i][j]);
			//printf("%d ",*((*p+i)+j));
		}
	}
	printf("\n");
}
void print_arr2(int arr[3][5],int row,int col)
{
	int i = 0;
	for (i = 0; i < row ; i++)
	{
		int j = 0;
		for (j = 0; j < col; j++)
		{
			printf("%d ", arr[i][j]);
		}
	}
	printf("\n");
}
int main()
{
	int arr[3][5] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };
	print_arr1(arr, 3, 5);
	//数组名arr表示的是数组首元素的地址,二维数组的首元素是指二维数组的第一行
	//这里传参的arr其实是指二维数组第一行的地址,是一个一维数组的地址。可以用数组指针来接收
	print_arr2(arr, 3, 5);
	return 0;
}

二、函数指针

1、理解函数指针和定义
函数名是函数的地址,(test())
&函数名也是函数的地址(
&test())

void test()
{
	printf("hehe\n");
}
int main()
{
	printf("%p\n", test);//00FA11E5
	printf("%p\n", &test);//00FA11E5
	return 0;

}
int Add(int x, int y)
{
	return x + y;
}
int main()
{
	int(*p)(int, int) = &Add;
	printf("%d\n", p(1, 2));//输出3,p为指向函数的指针,类型为int(*)(int,int)
	return 0;
}

这里我们来看两个有趣的定义方式:

 (*(void(*)())0)()

理解:1、void(* )()指的是一个返回类型为void的函数指针类型
2、(void(* )())0指的是将0强制类型转换为函数指针类型
3、( * (void (*) () ) 0 )()指的是调用0地址处无参数返回类型为void的函数

void (*signal(int ,void(*)(int)))(int)

理解:1、signal(int ,void( * )(int))是指一个函数,参数是一个int类型和一个函数指针类型
2、signal的返回类型为void(*)(int)
signal是一个函数申明,函数的参数有两个,第一个为int,第二个为函数指针,
该指针指向的函数有一个参数,类型为int,函数的返回类型为void,signal的返回类型为函数指针类型
该指针指向的函数有一个参数,类型为int,函数的返回类型为void

void (*signal(int ,void(*)(int)))(int)的改造
typedef void(*pfun_t)(int)//名字必须与*在一起
pfun_t signal(int,pfun_t)//pfun_t指的是void(*)(int)

三、函数指针数组、转移表

1、理解函数指针数组和定义,转移表
将函数的地址存到一个数组里,这个数组就叫做函数指针数组int (*parr1[10])()
函数指针数组的应用,转移表(计算器)

void Calc(int(*pfun)(int, int))
{
	int x = 0;
	int y = 0;
	int ret = 0;
	printf("请输入两个操作数:>");
	scanf("%d%d", &x, &y);
	ret = pfun(x, y);

}
enum Option
{
	EXIT,//0
	ADD,//1
	SUB,//2
	MUL,//3
	DIV//4
};
void menu()
{
	printf("*************************************\n");
	printf("****  1.Add          2.Sub       ****\n");
	printf("****  3.Mul          4.div       ****\n");
	printf("*************************************\n");
}
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 Cale(int(*pfun)(int,int))
{
	int x = 0;
	int y = 0;
	int ret = 0;
	printf("请输入两个操作数:");
	scanf("%d%d", &x, &y);
	ret = pfun(x, y);
	printf("ret = %d\n", ret);
}

int main()
{
	int input = 1;
	int ret = 0;
	do
	{
		menu();
		printf("请选择:");
		scanf("%d", input);
		switch (input)
		{
		case ADD:
			Cale(Add);
			break;
		case SUB:
			Cale(Sub);
			break;
		case MUL:
			Cale(Mul);
			break;
		case DIV:
			Cale(Div);
			break;
		case 0:
			printf("退出程序\n");
			break;
		default:
			printf("输入错误,请重新输入!");
			break;
		}

	} while (input);
	return 0;
}
int main()
{
	int x, y;
	int input = 1;
	int ret = 0;
	int(*pfun[5])(int x, int y) = { 0, Add, Sub, Mul, Div };
	while (input)
	{
		menu();
		printf("请选择:");
		scanf("%d", &input);
		if (input >= 1 && input <= 4)
		{
			printf("请输入两个操作数:>");
			scanf("%d%d", &x, &y);
			int ret = pfun[input](x, y);
			printf("ret = %d\n", ret);
		}
		else
		{
			printf("输入错误,请重新输入!");
		}
	} 
	return 0;
}

四、指向函数指针数组的指针

1、理解指向函数指针数组的指针和定义

指向函数指针数组的指针是一个指针,指向一个数组,这个数组里存储的是函数指针

void test(const char* str)
{
	printf("%s\n", str);
}
int main()
{
	//函数指针
	void(*pfun)(const char*) = test;
	//函数指针数组
	void(*ppfun[5])(const char*);
	ppfun[0] = test;
	//指向函数指针数组的指针
	void(*(*ppfun_t)[5])(const char*) = &ppfun;
}
char* test1(int a, int* b)
{
	printf("haha\n");
}
char* test2(int a, int* b)
{
	printf("hehe\n");
}
int main()
{
	//函数指针
	char* (*pfun)(int, int*) = test1;
	//函数指针数组
	char* (*pfun_t[10])(int, int*) = { test1, test2 };
	//指向函数指针数组的指针
	char* (*(*ppfun_t)[10])(int, int*) = &pfun_t;
}

五、回调函数

1、理解回调函数的使用

回调函数就是一个通过函数指针调用的函数,如果把函数的指针(地址)作为参数
传递给另一个函数,当这个指针被用来调用其所指向的函数时,这就叫做回调函数。

六、qsort函数

1、练习使用qsort函数排序各种类型的数据。
qsort函数的使用

#include<stdlib.h>
int int_cmp(const void* p1, const void* p2)
{
	return(*(int *)p1 - *(int*)p2);
}
int main()
{
	int arr[] = { 1,4, 3, 2, 6, 5, 7, 8, 9, 10 };
	int i = 0;
	//void qsort(void *base, size_t num, size_t width, 
	//int(__cdecl *compare)(const void *elem1, const void *elem2));
	qsort(arr, sizeof(arr) / sizeof(arr[0]), sizeof(int), int_cmp);
	for (i = 0; i < sizeof(arr) / sizeof(arr[0]); i++)
	{
		printf("%d ", arr[i]);
	}
	printf("\n");
	return 0;
}
#include<stdlib.h>
#include<string.h>
int int_cmp(const void* p1, const void* p2)
{
	return(*(char *)p1 - *(char*)p2);
}
int main()
{
	char arr[] = "acbdfeqg";
	qsort(arr, strlen(arr), sizeof(char), int_cmp);
	printf("%s\n", arr);
	return 0;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值