对C中回调函数的一些理解(实现泛型排序函数)

 对C中回调函数的一些理解(实现泛型排序函数)

 回调函数 : 

回调函数就是一个通过函数指针调用的函数。如果你把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用来调用其所指向的函数时,我们就说这是回调函数。回调函数不是由该函数的实现方直接调用,而是在特定的事件或条件发生时由另外的一方调用的,用于对该事件或条件进行响应。

说到回调函数 , 就不得不提qsort()这个库函数 , 它的一个参数就是一个函数指针(地址) 
qsort()是快排函数:
void qsort(void * base,size_t num,size_t size, int(* compar)(const void *,const void *));
其中 
 base 指向数组的起始地址 , 通常传入数组名或数组指针
 size_t num 为数组长度 , (设数组名为arr)通常传sizeof(arr) / sizeof(arr[0])
 size_t size 为数组元素类型长度(字节数) , (假设数组元素类型为int型)通常传sizeof(int)
  int(* compar)(const void *,const void *)为指向比较函数的函数指针,决定了排序的顺序。compar()就为回调函数
  话不多说 , 上代码

#include<stdio.h>
#include<stdlib.h>
int cmp(const void* a,const void* b) {
	return *(int*)a - *(int*)b;
}
int main() {
	int arr[10] = { 6,10,8,1,5,3,4,7,2,9 };
	for (int i = 0; i < 10; ++i) {
		printf("%d\t", arr[i]);
	}
	printf("\n");
	qsort(arr, sizeof(arr) / sizeof(arr[0]), sizeof(int), cmp);//对数组arr进行排序(快排)
	for (int i = 0; i < 10; ++i) {
		printf("%d\t", arr[i]);
	}
	printf("\n");
	system("pause");
	return 0;
}

我们可以效仿库函数qsort()自己写一个通用的排序函数 , 排序方法我们用冒泡排序 , 代码如下 : 

#include<stdio.h>
#include<stdlib.h>
#include<string.h>
int cmp_int(const void* a, const void* b) {//整型排序
	return *(int*)a - *(int*)b;
}
int cmp_double(const void* a, const void* b) {//double型排序
	if (*(double*)a - *(double*)b > 0) {
		return 1;
	}
	else if (*(double*)a - *(double*)b < 0) {
		return -1;
	}
	else {
		return 0;
	}
}
int cmp_char(const void* a, const void* b) {//字符型排序
	return *(char*)a - *(char*)b;
}
int cmp_str(const void* a, const void* b) {//字符串类型排序
	return strcmp(*(char**)a, *(char**)b);
}
void swap(char* p, char* q, int m) {
	for (int i = 0; i < m; ++i, ++p, ++q) { //按字节交换
		*p = *p^*q;
		*q = *p^*q;
		*p = *p^*q;
	}
}
void bsort(void* a,int n,int m,int (*cmp)(const void*, const void*)) {
	for (int i = 0; i < n; ++i) {
		for (int j = 0; j < n - i - 1; ++j) {
			if (cmp((char*)a + m * j, (char*)a + m * (j + 1)) > 0) { 
            //只有将强制转化为char*后指针加几就是向后偏移几
				swap((char*)a + m * j, (char*)a + m * (j + 1), m);
			}
		}
	}
}
int main() {
	int a1[10] = { 6,10,8,1,5,3,4,7,2,9 };
	char a2[10] = { 'b','g','a','d','e','r','c','f','n','z' };
	double a3[10]= { 6.23,10.12,8.34,1.0,5.6,3.90,4.788,7.3456,2.098,9.903 };
	char str1[5] = "dacd";
	char str2[5] = "abcd";
	char str3[5] = "edvv";
	char*a4[3] = { str1,str2,str3 };
	//1
	printf("int a1[10]排序前:\n");
	for (int i = 0; i < 10; ++i) {
		printf("%d\t", a1[i]);
	}
	printf("\nint a1[10]排序后\n");
	bsort(a1, sizeof(a1) / sizeof(a1[0]), sizeof(a1[0]), cmp_int);
	for (int i = 0; i < 10; ++i) {
		printf("%d\t", a1[i]);
	}
	printf("\n\n");
	//2
	printf("char a2[10]排序前:\n");
	for (int i = 0; i < 10; ++i) {
		printf("%c\t", a2[i]);
	}
	printf("\nchar a2[10]排序后\n");
	bsort(a2, sizeof(a2) / sizeof(a2[0]), sizeof(a2[0]), cmp_char);
	for (int i = 0; i < 10; ++i) {
		printf("%c\t", a2[i]);
	}
	printf("\n\n");
	//3
	printf("double a3[10]排序前:\n");
	for (int i = 0; i < 10; ++i) {
		printf("%f  ", a3[i]);
	}
	printf("\ndouble a3[10]排序后\n");
	bsort(a3, sizeof(a3) / sizeof(a3[0]), sizeof(a3[0]), cmp_double);
	for (int i = 0; i < 10; ++i) {
		printf("%f  ", a3[i]);
}
	printf("\n\n");
	//4
	printf("char*a4[3]排序前:\n");
	for (int i = 0; i < 3; ++i) {
		printf("str%d=%s  ", i + 1, a4[i]);
	}
	printf("\nchar*a4[3]排序后\n");
	bsort(a4, sizeof(a4) / sizeof(a4[0]), sizeof(a4[0]), cmp_str);
	for (int i = 0; i < 3; ++i) {
		printf("str%d=%s  ", i + 1, a4[i]);
	}
	printf("\n\n");
	system("pause");
	return 0;
}

运行结果如下 :  

值得注意的是这里的swap()函数在排序各种类型的数据时通用 , 因为这里的swap()函数时是通过一个一个交换变量字节来实现的, 所以对于任何类型通用 , 例如交换int型使 , int型为4字节 , swap()函数会一个字节一个字节的进行交换 , 当4个字节都交换时 , 两个int型数据就交换完成 , double型也一样 , 只是交换8次(double型为8字节) , char型为1个字节, 只交换1次 , 交换几次由bsort()函数中的第三个参数(元素类型长度(字节数))控制

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值