指针专题(4)【qsort函数的概念和使用】

1.前言

上节我们学习了指针的相关内容,本节我们在有指针的基础的条件下学习一下指针的运用,那么废话不多说,我们正式进入今天的学习

2.回调函数

我们既然已经学习了指针的相关基础,那么我们此时就可以用指针来实现回调函数

而回调函数又是什么呢?

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

如果我们把一个函数的地址作为参数传递给第二个函数,当第二个函数被用来调用它所指向的第一个函数的时候,此时第一个函数就被称为回调函数;

我们来举一个例子,这样就会更通俗易懂:假如有A和B两个函数,如果把A的地址传递给B,当B被用来调用A函数的时候,此时A函数就是回调函数;

我们需要注意:回调函数并不是由函数的实现方直接调用的,而是在特定的情况下通过另外一方去调用的,也就是说,回调函数是一种机制,用于对特定的事件进行响应;

上一节我们学习了计算器的代码编写,我们就可以使用回调函数的知识对代码进行修改和优化

我们在使用回调函数改造前,我们通常会这样编写代码:

int add(int a, int b)
{
	return a + b;
}
int sub(int a, int b)
{
	return a - b;
}
int mul(int a, int b)
{
	return a * b;
}
int div(int a, int b)
{
	return a / b;
}
int main()
{
    int x, y;
    int input = 1;
    int ret = 0;
    do
    {
        printf("*************************\n");
        printf("  1:add           2:sub  \n");
        printf("  3:mul           4:div  \n");
        printf("  0:exit                 \n");
        printf("*************************\n");
        printf("请选择:");
        scanf("%d", &input);
        switch (input)
        {
        case 1:
            printf("输⼊操作数:");
            scanf("%d %d", &x, &y);
            ret = add(x, y);
            printf("ret = %d\n", ret);
            break;
        case 2:
            printf("输⼊操作数:");
            scanf("%d %d", &x, &y);
            ret = sub(x, y);
            printf("ret = %d\n", ret);
            break;
        case 3:
            printf("输⼊操作数: ");
            scanf("%d %d", &x, &y);
            ret = mul(x, y);
            printf("ret = %d\n", ret);
            break;
        case 4:
            printf("输⼊操作数:");
            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;
}

此时的代码重复率非常高,代码非常冗余

我们若想要对代码进行优化,我们需要注意以下几点:

1.我们要把相似代码抽象成函数

2.我们学习了函数指针以后,我们知道了:函数的调用可以使用函数名来调用也可以使用函数指针来调用;我们可以通过参数的差异来计算不同的逻辑

有了这些理论基础我们就可以对代码进行修改了:

#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <string.h>
int add(int a, int b)
{
	return a + b;
}
int sub(int a, int b)
{
	return a - b;
}
int mul(int a, int b)
{
	return a * b;
}
int div(int a, int b)
{
	return a / b;
}

void calc(int(*pf)(int, int))
{
    int x, y;
    int ret = 0;
    printf("输入操作数:");
    scanf("%d %d", &x, &y);
    ret = pf(x, y);
    printf("ret = %d\n", ret);
}
int main()
{
    int input = 1;
    do
    {
        printf("*************************\n");
        printf("  1:add           2:sub  \n");
        printf("  3:mul           4:div  \n");
        printf("  0:exit                 \n");
        printf("*************************\n");
        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");
            break;
        default:
            printf("选择错误\n");
            break;
        }
    } while (input);
    return 0;
}

我们这个方法就是把一个函数的地址传递给了另外一个函数,然后通过另外一个函数用函数指针去调用要使用的函数

3.qsort函数

我们刚才学习了回调函数,我们现在接着来学习一个和回调函数逻辑差不多的函数:qsort函数

qsort函数是一个用来排序的函数,它是一个库函数,可以直接拿来排序,其底层原理使用的是快速排序的方式

我们之前说过:函数排序的底层方式有很多,大致包括:选择排序、插入排序、冒泡排序、快速排序、希尔排序

我们先来思考一下qsort函数为什么被设计出来呢?它的存在相比较其他的函数有什么优势呢?

我们首先将qsort函数与冒泡排序做一个对比,我们先写出冒泡排序的代码:

我们来回顾一下冒泡排序的核心思想:冒泡排序是将两两相邻的元素进行比较,如果不满足顺序就交换

void bubble_sort(int arr[], int sz)
{
    //趟数
    int i = 0;
    for (i = 0; i < sz - 1; i++)
    {
        //一趟内部的两两比较
        int j = 0;
        for (j = 0; j < sz - 1 - i; j++)
        {
            if (arr[j] > arr[j + 1])
            {
                int tmp = arr[j];
                arr[j] = arr[j + 1];
                arr[j + 1] = tmp;
            }
        }
    }

}
void print_arr(int arr[], int sz)
{
    int i = 0;
    for (i = 0; i < sz; i++)
    {
        printf("%d ", arr[i]);
    }
    printf("\n");
}
int main(void)
{
    //冒泡排序,排序为升序
    int arr[] = { 9,8,7,6,5,4,3,2,1,0 };
    int sz = sizeof(arr) / sizeof(arr[0]);
    bubble_sort(arr, sz);
    print_arr(arr, sz);
    return 0;
}

写完代码以后我们可以发现冒泡排序的局限性:因为冒泡排序数据传入的时候已经确定了类型,冒泡排序只能排序一种类型的数据,如果排序的是整型 ,则无法排序字符串、结构体数据等;如果要用冒泡排序排序字符串有需要重新创造一个函数,这样就会导致代码重复率高,代码冗杂等问题

此时我们就可以了解一下qsort函数,qsort函数可以排序任意类型的数据,我们来看一下qsort函数的使用方法:

我们通过上图可以知道,qsort函数使用需要包含头文件<stdlib.h>,而且qsort函数的使用需要4个变量:

1.void* base,该变量是一个指针变量,指针指向的是待排序数组的第一个元素

2.size_t num,该变量表示的是待排序数组里面的元素个数

3.size_t size,该变量表示的是待排序数组里面每个元素的大小

4.int (*compar)(const void*, const void*),该变量是一个函数指针,该函数指向一个函数,这个函数的功能是用来比较两个元素的大小。这个概念有点抽象,我们逐步理解:

刚才我们编写了冒泡排序的代码,假设我们需要在原冒泡排序代码的基础上改造冒泡排序。让它能够排序任意类型的数据,我们来思考一下哪些地方需要修改?

我们在两个数据的比较方式这里不能直接用大于号或者小于号比较只有整型元素可以直接用>、<比较,字符串、结构体等元素是不能用>、<比较的,而是使用strcmp函数等其他方法比较

所以要想比较所有类型的数据,我们比较的代码就需要修改:

此时我们就不能直接比较,应该要把比较两个元素的方法封装成一个函数,然后把函数的地址传递给排序函数。如果传递的是整型比较函数的地址,那么就按照整型的比较方式比较;如果传递的是字符比较函数的的地址,那么就按照字符串的比较方式比较......

其实这种方法就是qsort的比较方法, qsort函数的第四个变量指向的就是两个元素的比较函数,所以我们要传入两个元素比较函数的地址这里的传入需要我们自己实现,因为qsort函数的实现者并不知道要排序的数据的类型,只有qsort函数的使用者知道要排序的数据的类型,所以需要我们自己提供两个元素的比较函数,然后把比较函数的地址传给qsort函数,qsort函数内部在比较的时候就会调用这个函数;

那么知道了qsort函数的原理,我们来写一串代码使用qsort函数排序整型数据:

我们要想用qsort函数排序整型数据,那么我们需要自己完成整型数据比较的函数,并将函数地址传给qsort函数:

我们可以看到,第四个变量的形式是:int (*compar)(const void*, const void*)

所以我们在写代码的时候就需要按照以上的形式来确定变量的类型,这样qsort函数才能顺利接收到函数

我们可以看到,compare函数的两个参数的类型都是const void*,此时p1中存放的是第一个要比较元素的地址,p2中存放的是第二个要比较元素的地址,如果p1指向的元素大于p2指向的元素,那么将返回一个大于0的数字;如果p1指向的元素小于p2指向的元素,那么将返回一个小于0的数字;如果p1指向的元素等于于p2指向的元素,那么将返回一个等于0的数字;

如果要对两个元素进行比较,不能够直接解引用,因为p1和p2两个变量的类型是void* ,是没有具体类型的,我们不能直接解引用,也不能进行+-整数的运算。此时我们就需要强制类型转换,转换完以后再进行解引用

有了这些理论基础,我们此时就可以写出代码如下:

int cmp_int(const void* p1, const void* p2)
{
    if (*(int*)p1 > *(int*)p2)
        return 1;
    else if (*(int*)p1 == *(int*)p2)
        return 0;
    else
        return -1;
}

现在我们就可以完成全代码的书写了:

void print_arr(int arr[], int sz)
{
    int i = 0;
    for (i = 0; i < sz; i++)
    {
        printf("%d ", arr[i]);
    }
    printf("\n");
}

int cmp_int(const void* p1, const void* p2)
{
    if (*(int*)p1 > *(int*)p2)
        return 1;
    else if (*(int*)p1 == *(int*)p2)
        return 0;
    else
        return -1;
}

void test01()
{
    int arr[] = { 9,8,7,6,5,4,3,2,1,0 };
    int sz = sizeof(arr) / sizeof(arr[0]);
    qsort(arr, sz, sizeof(arr[0]), cmp_int);
    print_arr(arr, sz);
}

int main(void)
{
    test01();
    return 0;
}

代码运行成功,编写成功

如果我们再仔细思考一下,我们可以发现比较函数的代码可以简化:

int cmp_int(const void* p1, const void* p2)
{
    return (*(int*)p1 - *(int*)p2);
}

我们可以直接让两个变量解引用的值做差,也可以完成该函数的功能,并且代码量更少,更加简洁

如果我们需要对元素进行降序比较,我们修改代码也同样很方便,我们只需要让后面的元素减去前面的元素就可以实现降序的功能:

void print_arr(int arr[], int sz)
{
    int i = 0;
    for (i = 0; i < sz; i++)
    {
        printf("%d ", arr[i]);
    }
    printf("\n");
}

int cmp_int(const void* p1, const void* p2)
{
    return (*(int*)p2 - *(int*)p1);
}

void test01()
{
    int arr[] = { 5,8,7,6,9,4,3,2,1,0 };
    int sz = sizeof(arr) / sizeof(arr[0]);
    qsort(arr, sz, sizeof(arr[0]), cmp_int);
    print_arr(arr, sz);
}

int main(void)
{
    test01();
    return 0;
}

既然我们刚才已经学习了使用qsort函数排序整型数据,我们接着来用qsort函数排序一下结构体

我们知道,结构体是不能直接用>、<号来比较的,我们先来定义一个结构体:

struct Stu
{
    char name[20];
    int age;
};

我们定义了一个学生的结构体,结构体成员为学生的姓名和年龄,我们若是要对结构体进行排序,那么我们就有两种排序方法:

1.我们可以通过年龄来排序

2.我们可以通过姓名来排序

我们先来创建一个数组用于存储结构体数据:

    struct Stu arr[3] = { {"zhangsan",20},{"lisi",35},{"wangwu",18} };

1.我们先写用姓名来排序的代码:

要想通过姓名来排序,我们需要使用到strcmp函数:

strcmp 是按照对应字符串的字符的ASCII码值进行比较的函数

我们发现strcmp的返回值非常适合我们自己写的比较函数,所以我们可以写出代码如下:

int cmp_stu_by_name(const void* p1, const void* p2)
{
    return strcmp(((struct Stu*)p1)->name, ((struct Stu*)p2)->name);
}

我们先要对p1、p2强制类型转换为结构体类型,再通过结构体成员访问操作符->来找到姓名,再使用strcmp函数进行比较

通过这些,我们就可以很轻易的写出代码:

struct Stu
{
    char name[20];
    int age;
};

int cmp_stu_by_name(const void* p1, const void* p2)
{
    return strcmp(((struct Stu*)p1)->name, ((struct Stu*)p2)->name);
}

void print_arr(struct Stu arr[], int sz)
{
    int i = 0;
    for (i = 0; i < sz; i++)
    {
        printf("%s ", arr->name);
        arr++;
    }
    printf("\n");
}

void test02()//结构体
{
    struct Stu arr[3] = { {"zhangsan",20},{"lisi",35},{"wangwu",18} };
    int sz = sizeof(arr) / sizeof(arr[0]);
    qsort(arr, sz, sizeof(arr[0]), cmp_stu_by_name);
    print_arr(arr, sz);
}




int main(void)
{
    test02();
    return 0;
}

2.接下来我们再来按年龄排序

(代码相似度很高,逻辑差不多,所以不作详解,请自主思考解题过程谢谢)

struct Stu
{
    char name[20];
    int age;
};



int cmp_stu_by_age(const void* p1, const void* p2)
{
    return (((struct Stu*)p1)->age - ((struct Stu*)p2)->age);
}

void print_arr(struct Stu arr[], int sz)
{
    int i = 0;
    for (i = 0; i < sz; i++)
    {
        printf("%d ", arr->age);
        arr++;
    }
    printf("\n");
}

void test02()//结构体
{
    struct Stu arr[3] = { {"zhangsan",20},{"lisi",35},{"wangwu",18} };
    int sz = sizeof(arr) / sizeof(arr[0]);
    qsort(arr, sz, sizeof(arr[0]), cmp_stu_by_age);
    print_arr(arr, sz);
}




int main(void)
{
    test02();
    return 0;
}

模拟实现qsort函数

刚才我们学习了qsort函数的概念以及使用,现在我们来模仿qsort函数来实现一个冒泡排序的函数,这个函数可以排列任意类型的数据

因为我们要接受任意类型的数据,所以我们接受参数的临时变量的类型就不能是int,我们就需要将变量类型改成void* ,可以用来接收任意类型的数据

此时函数的参数就可以直接修改成qsort函数里的参数:

void bubble_sort(void* base, size_t sz, size_t width, int (*cmp)(const void* p1, const void* p2));

我们可以不用修改趟数只需要改变比较数据的方式

            if (cmp()>0)

那么此时cmp函数里面应该存入什么参数呢?我们应该存入arr[j]和arr[j+1]的地址,此时我们不能直接用&找到地址,我们应该用什么方法来取到它们的地址呢?

如果我们要找到下标为0和下标为1处的地址,我们知道下标为0处的地址就是base,下表为1处的地址就为base处的地址跳过width个字节,因为base是void* 类型的地址,不能进行+-操作,所以此时我们就需要把base转换成char* 类型的变量,然后再+width

所以arr[j]处的地址就为 (char*)base+j*width,arr[j+1]处的地址就为 (char*)base+(j+1)*width

            if (cmp((char*)base+j*width,(char*)base+(j+1)*width)>0)

此时我们交换的具体代码也需要改变,我们封装一个函数Swap,用于交换

我们需要往Swap函数里面传入3个参数:

Swap(cmp((char*)base + j * width, (char*)base + (j + 1) * width), width);

有了这些理论基础,我们就能很轻易的写出代码如下:

结尾

本节我们学习了指针的相关运用,加强了对指针的理解,下一节是指针的最后一节,谢谢您的浏览!!!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/566771.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

如何在在wordpress安装百度统计

前言 看过我的往期文章的都知道&#xff0c;我又建了一个网站&#xff0c;这次是来真的了。于是&#xff0c;最近在查阅资料时发现&#xff0c;有一款免费的软件可以帮我吗分析网站数据。&#xff08;虽然我的破烂网站压根没人访问&#xff0c;但是能装上的都得上&#xff0c;…

python爬虫 - 爬取html中的script数据(爬取 zum.com新闻)

文章目录 1. 分析页面内容数据格式2. 使用re.findall方法&#xff0c;编写爬虫代码3. 使用re.search 方法&#xff0c;编写爬虫代码 1. 分析页面内容数据格式 &#xff08;1&#xff09;打开 https://zum.com/ &#xff08;2&#xff09;按F12&#xff08;或 在网页上右键 --…

免 Administrator 权限安装软件

以欧路词典为例, 从官网下载的安装包 https://www.eudic.net/v4/en/app/download 直接运行会弹出 UAC 提示需要管理员权限. 一个词典而已, 为啥要管理员权限呢? 答案是安装程序默认使用的安装路径是 C:\Program Files\ 这就不难理解了. 对于这种不需要其他额外权限的软件, 可以…

以赛促学、生态共建 | 软通动力子公司鸿湖万联成功举办基于x86架构的OpenHarmony应用生态挑战赛

近日&#xff0c;由开放原子开源基金会、央视网、江苏省工业和信息化厅、无锡市人民政府、江苏软件产业人才发展基金会、苏州工业园区、无锡高新区等共同承办&#xff0c;鸿湖万联参与共建的“基于x86架构的OpenHarmony应用生态挑战赛”决赛路演在无锡圆满落幕。本次挑战赛历时…

【THM】Linux Privilege Escalation(权限提升)-初级渗透测试

介绍 权限升级是一个旅程。没有灵丹妙药,很大程度上取决于目标系统的具体配置。内核版本、安装的应用程序、支持的编程语言、其他用户的密码是影响您通往 root shell 之路的几个关键要素。 该房间旨在涵盖主要的权限升级向量,并让您更好地了解该过程。无论您是参加 CTF、参加…

【C++学习】STL之空间配置器之一级空间配置器

文章目录 &#x1f4ca;什么是空间配置器✈STL 提供六大组件的了解&#x1f440;为什么需要空间配置器&#x1f44d;SGI-STL空间配置器实现原理&#x1f302;一级空间配置器的实现 &#x1f4ca;什么是空间配置器 空间配置器&#xff0c;顾名思义就是为各个容器高效的管理空间…

录制课程用什么软件好?这2款软件让你脱颖而出

在当今这个信息化快速发展的时代&#xff0c;录制课程已经成为了一种常见的教学手段。无论是高校教师、培训师还是网络教育工作者&#xff0c;都需要借助一些软件来录制高质量的课程。那么&#xff0c;录制课程用什么软件好呢&#xff1f;接下来&#xff0c;本文将介绍两种常见…

DxO Nik Collection 6.10.0 8套滤镜胶片插件套件mac/win

DxO Nik Collection 6是一款专为摄影师和图像创作者打造的强大后期处理工具。无论是专业摄影师还是业余爱好者&#xff0c;它都能为您的照片带来前所未有的提升。 这款软件集合了众多经典的Nik滤镜插件&#xff0c;如Color Efex Pro、Silver Efex Pro等&#xff0c;以及新增的P…

微信抽奖活动怎么做_微信抽奖大狂欢

随着科技的飞速发展&#xff0c;微信已经成为我们生活中不可或缺的一部分。它不仅仅是一个简单的通讯工具&#xff0c;更是一个集社交、购物、娱乐等多种功能于一体的平台。今天&#xff0c;我们为大家带来了一场别开生面的微信抽奖活动&#xff0c;让你在享受乐趣的同时&#…

Linux CentOS 7中Nginx 1.8.0 安装SSL证书

文章目录 前言一、创建一个文本文件&#xff0c;将每个证书的内容复制并粘贴到文件中二、将证书文件和私钥上传服务器三、编辑nginx的配置文件四、重新载入nginx配置文件五、使用浏览器访问自己的域名测试证书是否成功即可六、服务器证书的备份与恢复注意 前言 要在Nginx中安装…

面试题集中营—分布式共识算法

分布式共识算法目标 分布式主要就是为了解决单点故障。一开始只有一个服务节点提供服务&#xff0c;如下图所示。那么如果服务节点挂了&#xff0c;对不起等着吧。 为了服务的高可用性&#xff0c;我们一般都会多引入几个副节点当备份&#xff0c;当服务节点挂了&#xff0c;就…

羊大师:夏季羊奶的好处有哪些?

夏季羊奶的好处主要包括以下几点 1.增强免疫力&#xff1a;羊奶中的钙元素丰富&#xff0c;能有效为身体补充营养物质&#xff0c;增强自身的免疫能力。羊奶还富含上皮细胞生长因子&#xff08;EGF&#xff09;&#xff0c;对人体鼻腔、咽喉、血管、肠胃等黏膜有良好的修复作用…

day83 AJAX

1什么是AJAX AJAX语法 AJAX Asynchronous JavaScript and XML 异步js和XML 实现页面某一部份更新&#xff0c;无需服务器转发或重定向 1 $.ajax() 语法: $.ajax( { "url" : "url&qu…

Android11 SystemUI clock plugin 插件入门

插件的编写 参照ExamplePlugin&#xff0c;需要系统签名。 需要先编译以下模块得到jar&#xff0c;引用在项目中。 m SystemUIPluginLibcom.android.systemui.permission.PLUGIN PluginManager.addPluginListener SystemUI 是如何发现 clock plugin 的&#xff1f; Syste…

光纤网络电力控制系统设计方案:623-6U CPCI的光纤网络电力控制系统

6U CPCI的光纤网络电力控制系统 一、设备概述 柔性直流输电系统中用于控制与测量的FS系统&#xff0c;适用于风电和太阳能发电的并网快速数值计算和闭环控制&#xff0c;以及与直流输电系统的换流器有关的特殊控制功能&#xff0c;包括门控单元的信号处理。该控制板的最大…

仓储管理解决方案:混合低代码与定制开发,实现高灵活性与高效率

引言 在当今竞争激烈的商业环境中&#xff0c;仓储管理成为了企业供应链中不可或缺的一环。有效的仓储管理不仅可以帮助企业降低库存成本、提高库存周转率&#xff0c;还能够提升客户满意度和整体运营效率。然而&#xff0c;随着市场需求的不断变化和业务规模的不断扩大&#…

第⑯讲:Ceph集群Pool资源池管理以及PG的数据分布的核心技术要点

文章目录 1.Pool资源池的管理1.1.查看Pool资源池列表1.2.创建一个Pool资源池1.3.查看Pool资源池的参数信息1.4.修改Pool资源池的参数信息1.5.为Pool资源池设置应用模式1.6.重命名Pool资源池1.7.设置Pool资源池的限额1.8.删除Pool资源池1.9.查看Pool资源池的利用率 2.PG的数据分…

打印菱形(C语言)

一、N-S流程图&#xff1b; 二、运行结果&#xff1b; 三、源代码&#xff1b; # define _CRT_SECURE_NO_WARNINGS # include <stdio.h>int main() {//初始化变量值&#xff1b;int i 0;int k 0;int j 0;//打印上半部分&#xff1b;for (i 1; i < 4; i){//打印空…

【机器学习300问】78、都有哪些神经网络的初始化参数方法?

在训练神经网络时&#xff0c;权重初始化是确保良好收敛的关键步骤之一。不合适的初始化方法可能会导致梯度消失或爆炸&#xff0c;特别是在深层网络中。那么都有哪些神经网络的初始化参数方法呢&#xff1f;选择它这些方法的原则是什么&#xff1f; 一、常用神经网络初始化参…

Parallels Desktop 19完美中文版 PD19虚拟机详细图文安装教程 亲测兼容M1/M2

对于许多Mac用户来说&#xff0c;运行Windows应用程序是必不可少的。也许你的雇主使用的软件只适用于Windows&#xff0c;或者需要使用依赖于某些Windows技术的网站。或者你想在Mac上玩Windows游戏。或者&#xff0c;你可能需要在其他操作系统上测试应用程序和服务——你可以在…