欢迎来到尧图网

客户服务 关于我们

您的位置:首页 > 教育 > 高考 > 深入理解指针(4)

深入理解指针(4)

2025/5/2 11:39:55 来源:https://blog.csdn.net/khjjjgd/article/details/142149902  浏览:    关键词:深入理解指针(4)

一 , 回调函数是什么?

回调函数就是一个通过函数指针调用的函数。 

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

可以查看之前写过的博客《函数指针数组的应用 - 转移表》

函数指针数组的运用 -- 转移表-CSDN博客

 我们分析红色框里的代码,只有调用函数的逻辑是有差异的,我们可以把调用的函数的地址以参数的形式传递过去,使用函数指针接收函数指针指向什么函数就调用什么函数,这里其实使用的就是回调函数。

对代码进行优化:

void menu()
{printf("**************************\n");printf("***** 1.Add  2.Sub   *****\n");printf("***** 3.Mul  4.Div   *****\n");printf("*****   0. exit      *****\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 Calc(int(*pf)(int, int))
{int x = 0;int y = 0;int ret = 0;printf("请输入两个操作数:> ");scanf("%d %d", &x, &y);ret = pf(x, y);printf("%d\n", ret);printf("\n");
}
int main()
{int input = 0;do {menu();printf("请选择:>  ");scanf("%d", &input);switch (input){case 1:Calc(Add);break;case 2:Calc(Sub);break;case 3:Calc(Mul);break;case 4:Calc(Div);break;case 0:printf("退出计算机!\n");printf("\n");break;default:printf("输入错误!请重新输入!\n");printf("\n");break;}} while (input);return 0;
}

 

二 ,qsort 使用举例

2.1 经典题型回顾 

 在讲解qsort 函数之前 , 我们看一下使用冒泡排序对一个数组实现升降序排序:

void bubble_sort(int arr[], int sz)
{//一趟for (int i = 0; i < sz-1; i++){//一次排序for (int j = 0; j < sz - 1 - i; j++) {//这里可以选择升序或者降序if (arr[j] < arr[j + 1]){int temp = arr[j];arr[j] = arr[j + 1];arr[j + 1] = temp;}}}for (int i = 0; i < sz; i++){printf("%d ", arr[i]);}printf("\n");
}
int main()
{int arr[10] = { 2,5,4,6,7,8,9,3,1,0 };int sz = sizeof(arr) / sizeof(arr[0]);bubble_sort(arr, sz);return 0;
}

 然后我们不难发现,使用这种排序方法是具有局限性的,如果我需要比较结构体变量与字符串的话,需要做很大的改动。

2.2 qsort 函数的使用方法 

但是我们C语言的库函数中给我们提供了一个快速排序的函数 , qsort 函数 

(quick sort)

它可以实现任意类型数据的排序,快速排序,而qsort 函数底层就是使用了回调函数

使用方法是:

void qsort(void* base,    size_t num,     size_t size,    int (*compar)(const void*, const void*));

 使用时:要包含头文件<stdlib.h>

 

2.3  使用qsort 函数排序整型数据 

 void* 类型的指针不能解引用操作,也不能+/-的整数操作
用来干嘛呢?                      用来存放地址的
使用之前要强制类型转换成想要的类型 

 #include <stdlib.h>
int cmp_int(const void* e1, const void* e2)
{if (*(int*)e1 > *(int*)e2)return 1;else if (*(int*)e1 < *(int*)e2)return -1;elsereturn 0;
}void test1()
{int arr[10] = { 2,5,6,7,8,9,1,3,4,0 };int sz = sizeof(arr) / sizeof(arr[0]);//使用qsort 函数排序整型数组arrqsort(arr, sz, sizeof(arr[0]), cmp_int);for (int i = 0; i < sz; i++){printf("%d ", arr[i]);}printf("\n");
}
int main()
{test1();return 0;
}

#include <stdlib.h>
int cmp_int(const void* e1, const void* e2)
{return *(int*)e2 - *(int*)e1;
}void test1()
{int arr[10] = { 2,5,6,7,8,9,1,3,4,0 };int sz = sizeof(arr) / sizeof(arr[0]);//使用qsort 函数排序整型数组arrqsort(arr, sz, sizeof(arr[0]), cmp_int);for (int i = 0; i < sz; i++){printf("%d ", arr[i]);}printf("\n");
}
int main()
{test1();return 0;
}

2.4  使用qsort 函数排序结构数据 

cmp_stu_by_age 是用来比较两个结构体对象的
那么e1就指向一个结构体对象,e2也指向一个结构体对象 

比较年龄: 

#include <stdlib.h>
#include <string.h>
struct Stu
{char name[20];int age;
};int cmp_Stu_by_age(const void* e1, const void* e2)
{return (*(struct Stu*)e1).age - (*(struct Stu*)e2).age;
}void test2()
{struct Stu s[] = { {"zhangsan",26},{"lisi",16},{"wangwu",29} };int sz = sizeof(s) / sizeof(s[0]);qsort(s, sz, sizeof(s[0]), cmp_Stu_by_age);
}int main()
{test2();return 0;
}

 比较名字:

1. 是字符对应的位置进行比较

2.按照名字来比较大小的时候
   名字是字符串,不能直接使用>比较
   应该使用strcmp函数比较大小
   strcmp (字符串1,字符串2)
   如果字符串1 大于 字符串2  返回 >0的数字
   如果字符串1 等于 字符串2  返回 0
   如果字符串1 小于 字符串2  返回 <0的数字

使用strcmp 函数的时候,要包含头文件<string.h>

#include <stdlib.h>
#include <string.h>
struct Stu
{char name[20];int age;
};int cmp_Stu_by_name(const void* e1, const void* e2)
{return strcmp((*(struct Stu*)e1).name , (*(struct Stu*)e2).name);
}void test2()
{struct Stu s[] = { {"zhangsan",26},{"lisi",16},{"wangwu",29} };int sz = sizeof(s) / sizeof(s[0]);qsort(s, sz, sizeof(s[0]), cmp_Stu_by_name);
}int main()
{test2();return 0;
}

 

三 ,qsort 函数的模拟实现

 使用回调函数,模拟实现  qsort  (采用冒泡的方式)

为了模拟实现qsort 函数的功能,我们首先要分析 qsort 函数每一个参数为什么这么写,这么写的目的是什么?如果不这么写会导致什么?

 

#include <stdio.h>
int int_cmp(const void * p1, const void * p2)
{return (*( int *)p1 - *(int *) p2);
}
void _swap(void *p1, void * p2, int size)
{int i = 0;for (i = 0; i< size; i++){char tmp = *((char *)p1 + i);*(( char *)p1 + i) = *((char *) p2 + i);*(( char *)p2 + i) = tmp;}
}
void bubble(void *base, int count , int size, int(*cmp )(void *, void *))
{int i = 0;int j = 0;for (i = 0; i< count - 1; i++){for (j = 0; j<count-i-1; j++){if (cmp ((char *) base + j*size , (char *)base + (j + 1)*size) > 0){_swap(( char *)base + j*size, (char *)base + (j + 1)*size, 
size);}}}
}
int main()
{int arr[] = { 1, 3, 5, 7, 9, 2, 4, 6, 8, 0 };int i = 0;bubble(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;
}

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com

热搜词