C语言之深入指针及qsort函数(五)(详解介绍)

C语言之深入指针

在这篇博客看不懂的可以看看这篇C语言之深入指针(四)在上篇博客中介绍了:

  1. 函数指针变量
  2. 函数指针数组
  3. 简易计算器的实现\

文章目录

  • C语言之深入指针
    • 1 回调函数
    • 2 qsort函数的使用
      • 2.1 使用冒泡排序排序整型数组
      • 2.2 使用qsort函数排序整型数组
      • 2.2 使用qsort函数排序结构体数组
        • 2.2.1 按照年龄来排序
        • 2.2.2 按照名字来排序
      • 3 qsort函数总结

1 回调函数

使用回调函数修改前的代码:

#include <stdio.h>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;
}
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("请输入操作数:>>");scanf("%d %d", &x, &y);ret = Add(x, y);printf("%d\n", ret);break;case 2:printf("请输入操作数:>>");scanf("%d %d", &x, &y);ret = Sub(x, y);printf("%d\n", ret);break;case 3:printf("请输入操作数:>>");scanf("%d %d", &x, &y);ret = Mul(x, y);printf("%d\n", ret);break;case 4:printf("请输入操作数:>>");scanf("%d %d", &x, &y);ret = Div(x, y);printf("%d\n", ret);break;case 0:printf("退出计算器\n");break;default:printf("请重新输入\n");}} while (input);return 0;
}
case 1:printf("请输入操作数:>>");scanf("%d %d", &x, &y);ret = Add(x, y);printf("%d\n", ret);break;

在上述代码中,可以看到在每个case语句中,代码基本相似,基本逻辑是一致的,但是输入和输出部分完全一致这么写显得代码很冗余,那么我们可以使用回调函数来减少代码的重复

回调函数:回调函数就是⼀个通过函数指针调⽤的函数
如果你把函数的指针(地址)作为参数传递给另⼀个函数,当这个指针被⽤来调⽤其所指向的函数时,被调⽤的函数就是回调函数。回调函数不是由该函数的实现⽅直接调⽤,⽽是在特定的事件或条件发⽣时由另外的⼀⽅调⽤的,⽤于对该事件或条件进⾏响应

使用回调函数修改后的代码:

#include <stdio.h>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 cacl(int (*p) (int, int) )
{int x = 0;int y = 0;int ret = 0;printf("请输入操作数:>>");scanf("%d %d", &x, &y);ret = p(x, y);printf("%d\n", ret);
}
int main(){int input = 0;do{menu();printf("请选择:>>");scanf("%d", &input);switch (input){case 1:cacl(Add);break;case 2:cacl(Sub);break;case 3:cacl(Mul);break;case 4:cacl(Div);break;case 0:printf("退出计算器\n");break;default:printf("请重新输入\n");break;}} while (input);return 0;
}

在上述代码中,使用了回调函数 cacl ,将函数的地址传给了回调函数,在回调函数中使用了加法等函数,减少了代码的重复
cacl函数的形参部分为函数指针类型用来接收函数的地址

2 qsort函数的使用

2.1 使用冒泡排序排序整型数组

给定一个整型数组,要求将其排序
最简单的方法就是使用冒泡排序
代码如下:

#include <stdio.h>
void bobble_sort(int arr[], int sz)
{int i = 0;int tmp = 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]){tmp = arr[j];arr[j] = arr[j + 1];arr[j + 1] = tmp;}}}
}Print(int arr[], int sz)
{int i = 0;for (i = 0; i < sz; i++){printf("%d ", arr[i]);}printf("\n");
}
int main(){int arr[] = { 1,4,7,2,5,8,10,3,6,9 };int sz = sizeof(arr) / sizeof(arr[0]);Print(arr, sz);bobble_sort(arr, sz);Print(arr, sz);return 0;
}

2.2 使用qsort函数排序整型数组

在2.1中使用冒泡排序排序了整型数组,但是bobble_sort这个函数只能排序整型数组,在C语言中有这么一个函数——qsort可以排序任意类型的数组

void qsort (void* base,size_t num, size_t size,            int (*compar)(const void*,const void*));//qsort函数的声明

在这里插入图片描述
(图片转载至https://cplusplus.com/)
在cplusplus中介绍到,qsort函数有四个形参

  1. void* base   //base指向的数组中第一个的元素  void* 为泛型指针可以接收任意类型的指针
  2. size_t num  //base指向的数组中元素的个数(待排序数组中的元素个数)
  3. size_t size  //base指向的数组中元素的大小(单位是字节)
  4. int (compar)(const void,const void*)) //函数指针 - 指针指向的函数是用来比较数组中的2个元素的
  5. 4中函数指针的例子:int compar (const void* p1, const void* p2);

我将在代码中详细介绍:

#include <stdio.h>
#include <stdlib.h>    //使用qsort需要包含的头文件//int compar (const void* p1, const void* p2);
int sort_intarr(const void* p1, const void* p2)  //比较数组中两个元素 形参类型是固定的
{return (*(int*)p1 - *(int*)p2);
}Print(int arr[], int sz)  //打印输出
{int i = 0;for (i = 0; i < sz; i++){printf("%d ", arr[i]);}printf("\n");
}
int main()
{int arr[] = { 1,4,7,2,5,8,10,3,6,9 };int sz = sizeof(arr) / sizeof(arr[0]);   //计算数组元素个数Print(arr, sz);  //排序前打印一次qsort(arr,sz,sizeof(arr[0]), sort_intarr);Print(arr, sz); //排序前打印一次后return 0;
}
qsort(arr,sz,sizeof(arr[0]), sort_intarr);

一、
在上述代码中传给了qsort函数4个参数
1 arr为数组名,即传给了qsort第一个元素
2 sz为计算出的元素个数
3 sizeof(arr[0])为计算出一个元素的大小
4 sort_intarr为自己写的一个函数,用于比较两个元素

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

二、
在上述代码中定义了一个sort_intarr函数用来比较两个元素的大小
return ( * (int * )p1 - (int * )p2);
1 由于形参为void*类型的数据,需要转换成int类型的数据进行比较

TIPS:如果想降序排列可以将return ( * (int * )p1 - * (int * )p2);改为return ( * (int * )p2 - * (int * )p1);

2.2 使用qsort函数排序结构体数组

由于结构体中的数据类型很多,所以我们得按需求来实现函数

2.2.1 按照年龄来排序

代码如下:

#include <stdio.h>
#include <stdlib.h>struct Stu
{char name[20];int age;
};int cmp_struct_by_age(const void* p1, const void* p2)
{return ((struct Stu*)p1)->age - ((struct Stu*)p1)->age;//return (*(struct Stu*)p1).age - (*(struct Stu*)p1).age;//两段代码等价
}int main()
{struct Stu arr[] = { {"zhangsan",25},{"lisi",18} ,{"wangwu",30} };int sz = sizeof(arr) / sizeof(arr[0]);qsort(arr, sz, sizeof(arr[0]), cmp_struct_by_age);return 0;
}

一、
与排序整型数组一样,传入的参数都是那4种

1 arr为数组名,即传给了qsort第一个元素
2 sz为计算出的元素个数
3 sizeof(arr[0])为计算出一个元素的大小
4 cmp_struct_by_age为自己写的一个函数,用于比较两个元素

//两段代码等价
return ((struct Stu*)p1)->age - ((struct Stu*)p1)->age;
return (*(struct Stu*)p1).age - (*(struct Stu*)p1).age;

二、
1 由于是结构体数组,所以要将void*类型的数据强制转换成结构体类型
2 如果使用 ( -> ) 操作符结构体数组不使用解引用操作符来使用
3 要想使用 ( . ) 操作符的话,需要在前面加上解引用操作符

2.2.2 按照名字来排序

代码如下:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>   //使用strcmp函数需要包含的头文件struct Stu
{char name[20];int age;
};int cmp_struct_by_name(const void* p1, const void* p2)
{return strcmp(((struct Stu*)p1)->name, ((struct Stu*)p2)->name);//return strcmp((*(struct Stu*)p1).name, (*(struct Stu*)p2).name);//两段代码等价
}int main()
{struct Stu arr[] = { {"zhangsan",25},{"lisi",18} ,{"wangwu",30} };int sz = sizeof(arr) / sizeof(arr[0]);qsort(arr, sz, sizeof(arr[0]), cmp_struct_by_name);return 0;
}

一、
与排序整型数组一样,传入的参数都是那4种

1 arr为数组名,即传给了qsort第一个元素
2 sz为计算出的元素个数
3 sizeof(arr[0])为计算出一个元素的大小
4 cmp_struct_by_name为自己写的一个函数,用于比较两个元素

//两段代码等价
return strcmp(((struct Stu*)p1)->name, ((struct Stu*)p2)->name);
return strcmp((*(struct Stu*)p1).name, (*(struct Stu*)p2).name);

二、
两个字符串之间的比较不是比较字符串长度,而是比较字符的ASCII
例如:

char str1 = "abcdef";
char str2 = "abz";

第一个字母都是a相等的话会比较下一对字符,c的ASCII值小于g的ASCII,所以str2的长度大于str1,我们可以使用strcmp函数来比较字符串的大小

3 qsort函数总结

1 qsort可以排序任意类型的数据
2 使用qsort需要包含头文件 <stdlib.h>
3 使用qsort需要传4个参数
4 根据需要排序的数据传递不同的函数 按照参数和返回值实现
5 升序使用第一个形参 - 第二个形参 降序则反之

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

什么是脏读、不可重复读、幻读讲解

数据库隔离级别是数据库管理系统中一个重要的概念&#xff0c;它定义了事务之间的可见性和影响。在多用户并发访问数据库时&#xff0c;隔离级别能够确保事务之间的相互独立性&#xff0c;避免数据不一致的问题。本文将深入探讨三种常见的并发问题&#xff1a;脏读、不可重复读…

QTableWidget 设置列宽行高大小的几种方式及其他常用属性设置

目录 效果&#xff1a; 1.列宽、行高自动分配 2.固定值 3.随内容分配列宽 随内容分配行高 4.水平方向标签拓展剩下的窗口部分&#xff0c;填满表格 5.列宽是自动分配的&#xff0c;但是第一列可手动调整宽度&#xff0c;而表格整体的列宽仍是自动分配的。第二、三列办法调…

String字符串性能优化的几种方案

原创/朱季谦 String字符串是系统里最常用的类型之一&#xff0c;在系统中占据了很大的内存&#xff0c;因此&#xff0c;高效地使用字符串&#xff0c;对系统的性能有较好的提升。 针对字符串的优化&#xff0c;我在工作与学习过程总结了以下三种方案作分享&#xff1a; 一.优…

SpringBoot整合Redis使用基于注解的缓存

环境准备 注解 EnableCaching CacheConfig CacheConfig 提供了一种在类级别共享公共缓存相关设置的机制。 | 参数 | 作用 | | | — | — | — | | cacheNames | 使用在类上的默认缓存名称 | | | keyGenerator | 用于类的默认KeyGenerator的bean名称 | | | cacheManager | 自定…

【ARM Trace32(劳特巴赫) 使用介绍 2.1 -- TRACE32 Practice 脚本 cmm 脚本学习】

请阅读【ARM Coresight SoC-400/SoC-600 专栏导读】 上篇文章【ARM Trace32(劳特巴赫) 使用介绍 2 - Veloce 环境中使用trace32 连接 Cortex-M33】 下篇文章【ARM Trace32(劳特巴赫) 使用介绍 2.2 – TRACE32 进阶命令之 DIAG 弹框命令】 文章目录 1. TRACE32 Practice 语法1.…

Python | 机器学习之PCA降维

​&#x1f308;个人主页&#xff1a;Sarapines Programmer&#x1f525; 系列专栏&#xff1a;《人工智能奇遇记》&#x1f516;少年有梦不应止于心动&#xff0c;更要付诸行动。 目录结构 1. 机器学习之PCA降维概念 1.1 机器学习 1.2 PCA降维 2. PCA降维 2.1 实验目的 2…

如何在Jupyter Lab中安装不同的Kernel

❤️觉得内容不错的话&#xff0c;欢迎点赞收藏加关注&#x1f60a;&#x1f60a;&#x1f60a;&#xff0c;后续会继续输入更多优质内容❤️ &#x1f449;有问题欢迎大家加关注私戳或者评论&#xff08;包括但不限于NLP算法相关&#xff0c;linux学习相关&#xff0c;读研读博…

Jupyter Notebook的下载安装与使用教程_Python数据分析与可视化

Jupyter Notebook的下载安装与使用 Jupyter简介下载与安装启动与创建NotebookJupyter基本操作 在计算机编程领域&#xff0c;有一个很强大的工具叫做Jupyter。它不仅是一个集成的开发环境&#xff0c;还是一个交互式文档平台。对于初学者来说&#xff0c;Jupyter提供了友好的界…

linux实现SSH免密登录设置,以及shell脚本实现

原创/朱季谦 最近在搭建linux集群&#xff0c;做了SSH免密登录的设置&#xff0c;正好把过程记录一下&#xff1a; 一.用搭建好的两台虚拟机做演示&#xff0c;A机器&#xff1a;192.168.200.129&#xff0c;B机器&#xff1a;192.168.200.128 二.分别在两台机器上执行以下步…

机器人制作开源方案 | 守护一体化护耆卫士

作者&#xff1a;白玲玲、张硕、孔亚轩单位&#xff1a;兰州理工大学指导老师&#xff1a;毕广利 1. 场景调研 “探索者”平台是结合机械、电子、传感器、计算机软硬件、控制、人工智能和造型技术等众多的先进技术研发推出的专业型机器人设备原型设计工具&#xff0c;包含机构…

利用jquery对HTML中的名字进行替代

想法&#xff1a;将网页中经常要修改的名字放在一个以jquery编写的js文件中&#xff0c;如果需要修改名字&#xff0c;直接修改js文件中的名字即可。 新建name_07.html文件&#xff0c;写入下面的代码&#xff1a; <!DOCTYPE html> <html> <head><meta …

Excel Unix时间戳和日期时间格式的相互转换

时间戳转日期时间 ((A18*3600)/86400)DATE(1970,1,1) # 或 (A18*3600)/8640070*36519# 带格式化 TEXT((C18*3600)/8640070*36519,"yyyy-mm-dd hh:mm:ss")首先加8小时进行时区转换&#xff0c;然后转换成天数&#xff0c;再加上1970年1月1日&#xff0c;最后设置日期…

kafka分布式安装部署

1.集群规划 2.集群部署 官方下载地址&#xff1a;http://kafka.apache.org/downloads.html &#xff08;1&#xff09;上传并解压安装包 [zhangflink9wmwtivvjuibcd2e package]$ tar -zxvf kafka_2.12-3.3.1.tgz -C ../software/&#xff08;2&#xff09;修改解压后的文件…

休闲娱乐 - 挂耳咖啡

公司有一个小的茶歇间&#xff0c;平时去喝个咖啡、放松身心、锻炼下身体。咖啡机是现磨咖啡豆的&#xff0c;喝喝就习惯了。 而我旁边一位同事习惯每天早上来自己泡一杯挂耳咖啡&#xff0c;再配上牛奶&#xff0c;感觉挺高级的。 关于挂耳咖啡就查了一下资料&#xff0c;介绍…

DDR SDRAM 学习笔记

一、基本知识 1.SDRAM SDRAM : 即同步动态随机存储器&#xff08;Synchronous Dynamic Random Access Memory&#xff09;, 同步是指其时钟频率与对应控制器&#xff08;CPU/FPGA&#xff09;的系统时钟频率相同&#xff0c;并且内部命令 的发送与数据传输都是以该时钟为基准…

web环境实现一键式安装启动

部署的痛点 一般在客户环境安装web环境&#xff0c;少说需要花费1-2小时。一般需要安装jdk、nginx、mysql、redis等 等你接触到了inno setup &#xff0c;你有可能会节约更少的时间去部署。也有可能是一个不懂技术的人&#xff0c;都可以进行操作的。废话不多说&#xff0c;接…

Ansible 企业实战详解

一、ansible简介1. ansible是什么2.ansible的特点ansible的架构图 二、ansible 任务执行1、ansible 任务执行模式2、ansible 执行流程3、ansible 命令执行过程 二 .Ansible安装部署1.yum安装2.ansible 程序结构3、ansible配置文件查找顺序4、ansible配置文件5.ansible自动化配置…

yolov5模型代码怎么修改

yaml配置文件 深度乘积因子 宽度乘积因子 所有版本只有这两个参数的不同&#xff0c;s m l x逐渐加宽加深 各种类型层参数对照 backbone里的各层&#xff0c;在这里解析&#xff0c;只需要改.yaml里的各层参数就能控制网络结构 修改网络结构 第一步&#xff1a;把新加的模块…

C++打怪升级(十一)- STL之list

~~~~ 前言1. list是什么2. list接口函数的使用1. 构造相关默认构造n个val构造迭代器范围构造拷贝构造 2 赋值运算符重载函数2 析构函数3 迭代器相关begin 和 endrbegin 和rend 4 容量相关emptysize 5 元素访问相关frontback 6 修改相关push_backpop_backpush_frontpop_frontins…

二进制的形式在内存中绘制一个对象实例

一、引用类型实例的内存布局 从内存布局的角度来看&#xff0c;一个引用类型的实例由如下图所示的三部分组成&#xff1a;ObjHeader TypeHandle Fields。前置的ObjHeader用来缓存哈希值和同步状态&#xff0c;TypeHandle部分存储类型对应方法表&#xff08;Method Table&…