C語言 —— 此去經年夢浪蕩魂音 - 深入理解指針(卷四)

目錄

1.?回調函數

2.?qsort函數

2.1 使用qsort函數排序整型數據

2.2 使用qsort排序結構數據

2.3 使用冒泡排序模擬實現qsort函數


1.?回調函數

回調函數其實就是一個通過函數指針調用的函數,如果你把函數的指針作為參數傳遞給另一個函數,當這個指針被用來調用其所指向的函數時,被調用的函數就是回調函數

??

回調函數不是由該函數的實現方直接調用,而是在特定的事件或條件發生時由另外的一方調用的,用于對該事件或條件進行響應

?先寫一個使用函數指針數組實現的簡易計算器:

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;
}

?但是我們可以發現,這段代碼中有很多重復的代碼,我們可以使用回調函數來進行優化

#include <stdio.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 ret = 0;int x, y;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("*************************\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;
}

把每種情況的函數統一由一個函數管理,把這些函數以參數的形式傳給該函數,這些函數就被稱為回調函數


2.?qsort函數

qsort函數文檔鏈接:

??

qsort - C++ Referencehttps://legacy.cplusplus.com/reference/cstdlib/qsort/?kw=qsort頭文件為:

??

include <stdlib.h> 

qsort函數的作用:用來排序的庫函數,可以直接用來對數據進行排序,底層使用的是快速排序,qsort可以對任意類型進行排序

如果p1指向的元素小于p2,則返回小于0的數字
如果二者相等,則返回0
如果p1指向的元素大于p2,則返回大于0的數字

? ??
以上三種情況默認排序為升序,若想降序以上結論反轉即可

2.1 使用qsort函數排序整型數據

#include <stdio.h>
#include <stdlib.h>int int_cmp(const void* p1, const void* p2)
{return (*(int*)p1 - *(int*)p2);
}int main()
{int arr[] = { 1, 3, 5, 7, 9, 2, 4, 6, 8, 0 };int i = 0;qsort(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;
}

我們定義一個int_cmp函數用于qsort函數在對數組進行排序時能比較數組元素大小關系的函數作為參數

? ??

int_cmp函數接受兩個const void *類型的指針p1和p2指針類型使得qsort函數可以處理各種類型數據的排序,在函數內部,先將p1和p2這兩個指針強制轉換為int *類型指針(這里我們必須進行強制轉換為int*類型,不能使用void*類型),以便能解引用獲取到對應的整數,然后通過計算這兩個整數的差值并返回

return (*(int*)p1 - *(int*)p2);

2.2 使用qsort排序結構數據

#include <stdio.h>
#include <stdlib.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;
}//假設按照名字來?較
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", 20}, {"lisi", 30}, {"wangwu", 15} };int sz = sizeof(s) / sizeof(s[0]);qsort(s, sz, sizeof(s[0]), cmp_stu_by_age);
}//按照名字來排序
void test3()
{struct Stu s[] = { {"zhangsan", 20}, {"lisi", 30}, {"wangwu", 15} };int sz = sizeof(s) / sizeof(s[0]);qsort(s, sz, sizeof(s[0]), cmp_stu_by_name);
}int main()
{test2();test3();return 0;}

strcmp:用于比較兩個字符串的大小

? ? ?

return strcmp(((struct Stu*)e1)->name, ((struct Stu*)e2)->name);

? ?

strcmp函數文檔鏈接:

??

strcmp - C++ Referencehttps://legacy.cplusplus.com/reference/cstring/strcmp/?kw=strcmp

2.3 使用冒泡排序模擬實現qsort函數

#include <stdio.h>
#include <stdlib.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;
}

因為不知道接收數據的類型所以我們用char* 來一個字節一個字節移動,,這就保證了這個模擬的函數能夠接受各種類型的數據?


未完待續~

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/diannao/74401.shtml
繁體地址,請注明出處:http://hk.pswp.cn/diannao/74401.shtml
英文地址,請注明出處:http://en.pswp.cn/diannao/74401.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

Shiro框架漏洞攻略

漏洞原理&#xff1a;服務端在接收到?個Cookie時&#xff0c;會按照如下步驟進?解析處理&#xff1a;1.檢索RememberMe Cookie的值 2.進?Base64解碼 3.進?AES解碼 4.進?反序列化操作 在第4步中的調?反序列化時未進?任何過濾&#xff0c;進?可以導致出發遠程代碼執?漏…

Ceph集群2025(Squid版)導出高可用NFS集群(下集 )

本次主要對接K8S和傳統的一樣而已,比較簡單&#xff0c;不再過多講解 官網 https://github.com/kubernetes-csi/csi-driver-nfs/tree/master/chartshelm repo add csi-driver-nfs https://raw.githubusercontent.com/kubernetes-csi/csi-driver-nfs/master/charts helm pull c…

洛谷題單入門4-P5729 【深基5.例7】工藝品制作-python

輸入格式 第一行三個正整數 w,x,h。 第二行一個正整數 q。 接下來 q 行&#xff0c;每行六個整數 輸出格式 輸出一個整數表示答案。 三維數組直接標記 class Solution:staticmethoddef oi_input():"""從標準輸入讀取數據"""w, x, h map(…

亞馬遜云科技全面托管DeepSeek-R1模型現已上線

文章目錄 亞馬遜云科技全面托管DeepSeek-R1模型現已上線在Amazon Bedrock中開始使用DeepSeek-R1模型DeepSeek-R1現已可用 亞馬遜云科技全面托管DeepSeek-R1模型現已上線 亞馬遜云科技提供眾多免費云產品&#xff0c;可以訪問&#xff1a;亞馬遜云科技 截至1月30日&#xff0c;D…

IO模型種類

文章目錄 同步阻塞 I/O&#xff08;Blocking I/O&#xff0c;BIO&#xff09;同步非阻塞 I/O&#xff08;Non-blocking I/O&#xff0c;NIO&#xff09;I/O 多路復用&#xff08;I/O Multiplexing&#xff09;信號驅動 I/O&#xff08;Signal-driven I/O&#xff09;異步 I/O&a…

C語言入門教程100講(40)文件定位

文章目錄 1. 什么是文件定位?2. 文件指針3. 文件定位函數3.1 `fseek` 函數3.2 `ftell` 函數3.3 `rewind` 函數4. 示例代碼代碼解析:輸出結果:5. 常見問題問題 1:`fseek` 的 `offset` 參數可以為負數嗎?問題 2:如何判斷文件定位是否成功?問題 3:`rewind` 和 `fseek(file…

el-table折疊懶加載支持排序

el-table折疊懶加載支持排序 因為el-table懶加載的子節點是通過緩存實現的&#xff0c;如果想在展開的情況下直接刷新對應子節點數據&#xff0c;要操作el-table組件自身數據&#xff0c;否則不會更新 以排序功能為例 maps: new Map() //用于存儲子節點懶加載的數據// 加載子…

Off-Road-Freespace-Detection配置pytorch2.0.0

一、概述 在github上進行開源代碼搜索&#xff0c;發現了Off-Road-Freespace-Detection&#xff08;鏈接如下所示&#xff09;。這是對越野環境可通行區域的檢測&#xff0c;在經過測試之后&#xff0c;發現對自己有益。 GitHub - chaytonmin/Off-Road-Freespace-Detection: O…

ChatGPT降低論文AIGC重復率的提示詞合集(高效降重方法)

&#x1f4a1; 問題&#xff1a;寫完畢業論文后&#xff0c;查AIGC率過高&#xff0c;手動降重后仍然很高&#xff0c;該怎么辦&#xff1f; &#x1f4cc; 解決方案&#xff1a; 1?? 先查AIGC率&#xff08;找出AI生成的部分&#xff09; 2?? 用ChatGPT優化&#xff08;使…

【Spring 新特性全解析】

Spring 新特性全解析 引言 在當今 Java 企業級開發領域&#xff0c;Spring 框架無疑是中流砥柱般的存在。它以其強大的功能、高度的可擴展性和便捷的開發體驗&#xff0c;贏得了廣大開發者的青睞。隨著技術的不斷演進&#xff0c;Spring 也在持續更新迭代&#xff0c;帶來了一…

System.arraycopy 在音視頻處理中的應用

在音視頻開發領域&#xff0c;我們經常需要處理大量的數據&#xff0c;例如音頻 PCM 數據的傳輸、視頻幀的緩存等。在這些場景下&#xff0c;數據的復制與傳輸往往直接影響到應用的性能。Java 提供的 System.arraycopy 方法&#xff0c;在音視頻處理代碼中出現頻率非常高。本文…

fastapi+angular評論和回復

說明&#xff1a;fastapiangular評論和回復 效果圖: step1:sql show databases; DROP TABLE users; SHOW CREATE TABLE db_school.users; show tables; use db_school; SELECT * FROM db_school.jewelry_categories; CREATE DATABASE db_school; select *from users -- 用戶…

C++11QT復習 (三)

文章目錄 [toc]Day5-2 文件IO&#xff08;2025.03.24&#xff09;1. 緩沖區與刷新1.1 常見的緩沖刷新方式 2. 文件讀寫操作2.1 讀取文件2.2 寫入文件2.3 追加模式寫入2.3 完整代碼 3. 文件定位操作4. 字符串IO5. 配置文件解析示例6. 完整代碼7. 二進制文件操作總結 Day5-2 文件…

Redis Sentinel 詳解

Redis Sentinel 詳解 1. 什么是 Redis Sentinel&#xff1f;有什么用&#xff1f; Redis Sentinel&#xff08;哨兵&#xff09; 是 Redis 官方提供的高可用性解決方案&#xff0c;主要用于監控、通知和自動故障轉移。當 Redis 主節點&#xff08;master&#xff09;發生故障…

AI日報 - 2025年3月25日

&#x1f31f; 今日概覽&#xff08;60秒速覽&#xff09; ▎&#x1f916; AGI突破 | Nebula&#xff08;Google Gemini 2.0 Pro&#xff09;破解復雜數學謎題 編碼與推理能力再上新臺階 ▎&#x1f4bc; 商業動向 | Sesame AI開源10億參數語音模型CSM-1B 語音AI進入普惠時代 …

AI醫療革命:英偉達GTC 2025醫療健康與生命科學會議全分析

AI醫療革命:英偉達GTC 2025醫療健康與生命科學會議全分析 一、GTC 2025:AI 醫療的算力與生態雙突破 1.1 黃仁勛演講核心:從訓練到推理的代際跨越 在科技界矚目的英偉達 GTC 2025 大會上,英偉達 CEO 黃仁勛的主題演講成為全場焦點,為 AI 醫療領域帶來了極具變革性的消息。…

【機器學習/大模型/八股文 面經 (一)】

1. PPO算法中使用GAE的好處以及參數γ和λ的作用是什么? 參考答案: GAE(Generalized Advantage Estimation) 的優勢在于通過指數加權多步TD誤差,平衡優勢估計的偏差與方差,提升策略優化的穩定性。γ(折扣因子):控制未來獎勵的衰減程度,值越大表示更關注長期收益。λ…

03 Python 基礎:數據類型、運算符與流程控制解析

文章目錄 一、數據類型 內置的六大類數字類型整數類型 int浮點數 float布爾 bool字符串 str 變量命名 二、數字類型的相互轉換顯式類型的轉換整數&#xff0c;浮點數&#xff0c;復數 之間的顯式轉換 隱式類型的轉換 三、標識符算術運算符比較運算符邏輯運算符位運算符賦值運算…

視頻知識庫初步設想

將視頻字幕提取出來作為知識庫來源定位,下一步設想:把視頻上的圖片信息也精簡出來作為定位。 下面是測試例子: 入參: {"model":"deepseek-ai/DeepSeek-R1-Distill-Llama-8B","messages":[{"role":"system","cont…

數據庫原理13

1.關系模式設計不當引起的問題&#xff1a;數據冗余&#xff1b;更新異常&#xff1b;插入異常&#xff1b;刪除異常 2.外碼可以是單個屬性&#xff0c;也可以是屬性組 3.動態SQL是SQL標準提供的一種語句運行機制 4.若一個模式分解保持函數依賴&#xff0c;則該分解一定具有…