郑州正规网站制作公司/手机如何做网站

郑州正规网站制作公司,手机如何做网站,wordpress文章归档页面,游戏代理怎么做一. 回调函数 1. 什么是回调函数? 回调函数(Callback Function)是通过 函数指针 调用的函数。其本质是: 将函数作为参数传递给另一个函数,并在特定条件下被调用,实现 反向控制。 2. 回调函数的使用 回调函…

一. 回调函数

  1. 什么是回调函数?

回调函数(Callback Function)是通过 函数指针 调用的函数。其本质是:
将函数作为参数传递另一个函数,并在特定条件下被调用,实现 反向控制

  2. 回调函数的使用

回调函数的主要用途包括事件处理和排序算法。通过传递函数指针作为参数,可以实现代码的灵活性可重用性。例如,在一个简单的计算器程序中,可以通过传递不同的计算函数(如加法、减法等)来简化代码。

  3. 示例代码

以下是一个简单的回调函数示例,展示了如何使用回调函数进行加法和减法操作:

#include <stdio.h>// 定义回调函数类型
typedef int (*Operation)(int, int);// 定义加法函数
int add(int a, int b) 
{return a + b;
}// 定义减法函数
int jian(int a, int b) 
{return a - b;
}// 使用回调函数进行计算
int calculate(int a, int b, Operation op){return op(a, b);
}int main() 
{int x = 10, y = 5;printf(" %d\n", calculate(x, y, add));printf(" %d\n", calculate(x, y, jian));return 0;
}

二.    qsort函数

1 qsort 函数的定义

qsort 函数是C语言标准库中的一个排序函数,用于对数组进行快速排序。它的完整声明如下:

#include <stdlib.h>
void qsort(void *base, size_t nmemb, size_t size,int (*compar)(const void * e1, const void * e2));
  • base:指向要排序的数组的第一个元素的指针。
  • nmemb:数组中元素的个数。
  • size:数组中每个元素的大小,以字节为单位。
  • compar:用来比较两个元素的函数,即函数指针(回调函数)。

2.compar参数

  • compar参数指向一个比较两个元素的函数。比较函数的原型应该像下面这样。注意两个形参必须是const void *型,同时在调用compar 函数(compar实质为函数指针,这里称它所指向的函数也为compar)时,传入的实参也必须转换成const void *型。在compar函数内部会将const void *型转换成实际类型。
  • int compar(const void *e1, const void *e2);
  • 如果compar返回值小于0(< 0),那么e1所指向元素会被排在e2所指向元素的左面
  • 如果compar返回值等于0(= 0),那么e1所指向元素与e2所指向元素的顺序不确定;
  • 如果compar返回值大于0(> 0),那么e1所指向元素会被排在e2所指向元素的右面

这里再强调一下 

   参数类型:必须为   const void*

参考代码: 

以下是一个使用 qsort 函数对整型数组进行升序排序的示例:

qsort  一般默认是升序

#include <stdio.h>
#include <stdlib.h>int comp(const void* e1, const void* e2)
{return (*(int*)e1 - *(int*)e2);
}
int main()
{int arr[6] = { 44,0,520,1314,888,444};int sz = sizeof(arr) / sizeof(arr[0]);qsort(arr, sz, sizeof(arr[0]), comp);/*for (int i = 0; i < sz; i++)printf("%d ", arr[i]);*/return 0;
}

运行结果: 

以下是一个使用 qsort 函数对结构体数组进行排序的示例, 

#include<stdio.h>
#include<stdlib.h>
#include<string.h>
typedef struct student
{int id;char name[10];int grade; 
}student;int cmp1(const void *a, const void *b)//一级排序
{student *s1 = (student*)a;student *s2 = (student*)b;return s1->id - s2->id;
}int cmp2(const void *a,const void *b)//二级排序
{student *s1 = (student*)a;student *s2 = (student*)b;if(strcmp(s1->name , s2->name) != 0)return strcmp(s1->name , s2->name);  else   return s1->id - s2->id;              
}int cmp3(const void *a,const void *b)//三级排序
{student *s1 = (student*)a;student *s2 = (student*)b;if(s1->grade != s2->grade)   return s1->grade - s2->grade;else{if(strcmp(s1->name , s2->name) != 0)return strcmp(s1->name , s2->name);elsereturn s1->id - s1->id;}
}int main()
{int i,N,C;scanf("%d %d",&N,&C);student *stu;stu=(student*)malloc(N*sizeof(student));for(i = 0 ; i < N ; i++)scanf("%d %s %d" , &stu[i].id , stu[i].name , &stu[i].grade);switch(C){case 1: qsort(stu, N, sizeof(student), cmp1);break;//一级排序case 2: qsort(stu, N, sizeof(student), cmp2);break;//二级排序case 3: qsort(stu, N, sizeof(student), cmp3);break;//三级排序}printf("排序结果:\n");for(i = 0 ; i < N ; i++)printf("%03d %s %d\n" , stu[i].id , stu[i].name , stu[i].grade);return 0;
}

三 .调试技巧与常见陷阱

1. 调试建议

  • 在比较函数中添加打印语句:

  • int compare_debug(const void *a, const void *b){ 
    int x = *(int*)a; 
    int y = *(int*)b;printf("Comparing %d vs %d\n", x, y); 
    return x - y; 
    }

2. 常见错误


掌握回调函数与qsort的配合使用,不仅能写出更通用的代码,更能深入理解C语言"函数即数据"的哲学思想。这种设计模式在文件操作、事件处理、算法策略等场景中广泛应用,是提升编程能力的重要阶梯。

  • 错误1:忘记类型转换

    // 错误写法: return *a - *b; 
    // a和b是void指针!// 正确: return *(int*)a - *(int*)b;

  • 错误2:整数溢出

  •  int compare_unsafe(const void *a, const void *b) 
    {return *(int*)a - *(void*)b;// 可能溢出! 
    }// 改进版:int compare_safe(const void *a, const void *b) 
    { 
    int x = *(int*)a; 
    int y = *(int*)b; 
    return (x > y) ? 1 : ((x < y) ? -1 : 0); 
    }

  • 错误3:修改const数据

    int compare_wrong(const void *a, const void *b) 
    {*(int*)a = 10;// 尝试修改原始数据!return ...;}
    四.最佳实践总结
  • 严格遵循比较函数规范
    确保返回正确的-1/0/1,而不仅仅是差值

  • 优先使用const修饰
    比较函数参数应声明为const void*

  • 复杂结构预先处理
    对频繁排序的大型结构,可建立索引数组

  • 跨平台注意字节序
    处理网络传输数据时考虑大小端问题

  • 性能关键处慎用
    高频调用场景可考虑特定优化

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

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

相关文章

vscode记录

vs code 下载安装&#xff0c;git 配置&#xff0c;插件安装_vscode安装git插件-CSDN博客 手把手教你在VS Code中使用 Git_vscode如何输入git命令-CSDN博客 VS Code | 如何快速重启VS Code&#xff1f;_vscode 怎么一键全部重启-CSDN博客 1&#xff0c;安装插件与git集成 2&am…

大数据学习(80)-数仓分层

&#x1f34b;&#x1f34b;大数据学习&#x1f34b;&#x1f34b; &#x1f525;系列专栏&#xff1a; &#x1f451;哲学语录: 用力所能及&#xff0c;改变世界。 &#x1f496;如果觉得博主的文章还不错的话&#xff0c;请点赞&#x1f44d;收藏⭐️留言&#x1f4dd;支持一…

数智读书笔记系列021《大数据医疗》:探索医疗行业的智能变革

一、书籍介绍 《大数据医疗》由徐曼、沈江、余海燕合著&#xff0c;由机械工业出版社出版 。徐曼是南开大学商学院副教授&#xff0c;在大数据驱动的智能决策研究领域颇有建树&#xff0c;尤其在大数据驱动的医疗与健康决策方面有着深入研究&#xff0c;曾获天津优秀博士论文、…

SpringSecurity——前后端分离登录认证

SpringSecurity——前后端分离登录认证的整个过程 前端&#xff1a; 使用Axios向后端发送请求 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><title>登录</title><script src"https://cdn…

qt下载和安装教程国内源下载地址

qt不断在更新中&#xff0c;目前qt6日渐成熟&#xff0c;先前我们到官方下载或者国内镜像直接可以下载到exe文件安装&#xff0c;但是最近几年qt官方似乎在逐渐关闭旧版本下载通道&#xff0c;列为不推荐下载。但是qt5以其广泛使用和稳定性&#xff0c;以及积累大量代码使得qt5…

Mysql架构理论部分

Mysql架构是什么&#xff1f;实际可以理解为执行一条sql语句所要经历的阶段有哪些&#xff01; 1.连接层 &#xff08;1&#xff09;客户端发起连接 客户端通过TCP/IP、Unix Socket或命名管道等方式向Mysql服务器发起链接请求 想要了解tcp与udp的区别&#xff0c;可以参考这…

架构师面试(十九):IM 架构

问题 IM 系统从架构模式上包括 【介绍人模式】和 【代理人模式】。介绍人模式也叫直连模式&#xff0c;消息收发不需要服务端的参与&#xff0c;即客户端之间直连的方式&#xff1b;代理人模式也叫中转模式&#xff0c;消息收发需要服务端进行中转。 下面关于这两类模式描述的…

【服务器】RAID0、RAID1、RAID5、RAID6、RAID10异同与应用

目录 ​编辑 一、RAID概述 1.1 磁盘阵列简介 1.2 功能 二、RAID级别 2.1 RAID 0&#xff08;不含校验与冗余的条带存储&#xff09; 2.2 RAID1&#xff08;不含校验的镜像存储&#xff09; 2.3 RAID 5 &#xff08;数据块级别的分布式校验条带存储&#xff09; 4、RAI…

MySQL身份验证的auth_socket插件

在Ubuntu 20.04 LTS上&#xff0c;MySQL 8.0默认使用auth_socket插件进行身份验证&#xff0c;可能存在意想不到的情况。 一、auth_socket插件 在使用sudo mysql或通过sudo切换用户后执行任何MySQL命令时&#xff0c;不需要输入密码或错误密码都可以正常登入mysql数据库&…

使用Python在Word中创建、读取和删除列表 - 详解

目录 工具与设置 Python在Word中创建列表 使用默认样式创建有序&#xff08;编号&#xff09;列表 使用默认样式创建无序&#xff08;项目符号&#xff09;列表 创建多级列表 使用自定义样式创建列表 Python读取Word中的列表 Python从Word中删除列表 在Word中&#xff…

软考-软件设计师-计算机网络

一、七层模型 中继器&#xff1a;信号会随着距离的增加而逐渐衰减&#xff0c;中继器可以接受一端的信息再将其原封不动的发给另一端&#xff0c;起到延长传输距离的作用&#xff1b; 集线器&#xff1a;多端口的中继器&#xff0c;所有端口公用一个冲突域&#xff1b; 网桥&…

关于Flask框架30道面试题及解析

文章目录 基础概念1. 什么是Flask?其核心特性是什么?2. Flask和Django的主要区别?3. 解释Flask中的“路由”概念。如何定义动态路由?核心组件4. Flask的请求上下文(Request Context)和应用上下文(Application Context)有什么区别?5. 如何访问请求参数?POST和GET方法的…

C++20 中 `constexpr` 的强大扩展:算法、工具与复数库的变革

文章目录 一、constexpr 在 <algorithm> 中的应用1. 编译时排序2. 编译时查找 二、constexpr 在 <utility> 中的应用1. 编译时交换2. 编译时条件交换 三、constexpr 在 <complex> 中的应用1. 编译时复数运算 四、总结 C20 对 constexpr 的增强是其最引人注目…

conda create之后,以前的conda env list 只能看到环境路径 没有环境名称了

1.命令 conda env list 看到的显示如下&#xff1a; 左边这列的"base"&#xff0c;指向的路径和其它环境变量安装的路径不一致。 这时需要通过"activate [anaconda的环境路径]"和"source activate"回到anaconda&#xff1a; 2.执行切换命令 …

vulnhub-Tr0ll ssh爆破、wireshark流量分析,exp、寻找flag。思维导图带你清晰拿到所以flag

vulnhub-Tr0ll ssh爆破、wireshark流量分析&#xff0c;exp、寻找flag。思维导图带你清晰拿到所以flag 1、主机发现 arp-scan -l 2、端口扫描 nmap -sS -sV 192.168.66.185 nmap -sS -A -T4 -p- 192.168.66.185 nmap --scriptvuln 192.168.66.185经典扫描三件套&#xff0c;…

强化学习(赵世钰版)-学习笔记(8.值函数方法)

本章是算法与方法的第四章&#xff0c;是TD算法的拓展&#xff0c;本质上是将状态值与行为值的表征方式&#xff0c;从离散的表格形式&#xff0c;拓展到了连续的函数形式。 表格形式的优点是直观&#xff0c;便于分析&#xff0c;缺点是数据量较大或者连续性状态或者行为空间时…

笛卡尔轨迹规划之齐次变换矩阵与欧拉角、四元数的转化

一、笛卡尔轨迹规划需求 笛卡尔轨迹规划本质就是我们对机械臂的末端位置和姿态进行规划&#xff0c;其实也就是对末端坐标系的位姿进行规划。我们清楚末端坐标系的位姿是可以用齐次变换矩阵T来表示的&#xff0c;但这样表示的话&#xff0c;并不利于我们去做规划&#xff0c;所…

大摩闭门会:250321 学习总结报告

如果图片分辨率不足&#xff0c;可右键图片在新标签打开图片或者下载末尾源文件进行查看 本文只是针对视频做相应学术记录&#xff0c;进行学习讨论使用

深入理解 Spring Boot 应用的生命周期:从启动到关闭的全流程解析

引言 Spring Boot 是当今 Java 开发中最流行的框架之一&#xff0c;它以简化配置和快速开发著称。然而&#xff0c;要真正掌握 Spring Boot&#xff0c;理解其应用的生命周期是至关重要的。本文将深入探讨 Spring Boot 应用的生命周期&#xff0c;从启动到关闭的各个阶段&…

期刊分区表2025年名单下载(经济学、管理学)

2025年期刊分区表包括SCIE、SSCI、A&HCI、ESCI和OAJ&#xff0c;共设置了包括自然科学、社会科学和人文科学在内的21个大类 本次分享的是期刊分区表2025年名单经济学类、管理学类&#xff0c;一共7631025条 一、数据介绍 数据名称&#xff1a;期刊分区表2025年名单 数据…