C語(yǔ)言qsort函數(shù)用冒泡排序?qū)崿F(xiàn)過程詳解
前言
這篇文章就是指針進(jìn)階的收尾環(huán)節(jié)了,相信看過C語(yǔ)言進(jìn)階——指針(下)的小伙伴一定還記著文章末尾的回調(diào)函數(shù)吧。這篇文章就是借qsort函數(shù)的模擬實(shí)現(xiàn)來(lái)給小伙伴們展示一下回調(diào)函數(shù)的運(yùn)用。
1.冒泡排序的實(shí)現(xiàn)
在實(shí)現(xiàn)qsort函數(shù),相信有的小伙伴對(duì)冒泡排序還存有疑惑,甚至是第一次接觸這個(gè)名詞,那么我們就先來(lái)講講冒泡排序。
1.1冒泡排序的概念
“冒泡排序(Bubble Sort),是一種計(jì)算機(jī)科學(xué)領(lǐng)域的較簡(jiǎn)單的排序算法。它重復(fù)地走訪過要排序的元素列,依次比較兩個(gè)相鄰的元素,如果順序(如從大到小、首字母從Z到A)錯(cuò)誤就把他們交換過來(lái)。走訪元素的工作是重復(fù)地進(jìn)行,直到?jīng)]有相鄰元素需要交換,也就是說(shuō)該元素列已經(jīng)排序完成。這個(gè)算法的名字由來(lái)是因?yàn)樵叫〉脑貢?huì)經(jīng)由交換慢慢“浮”到數(shù)列的頂端(升序或降序排列),就如同碳酸飲料中二氧化碳的氣泡最終會(huì)上浮到頂端一樣,故名“冒泡排序”。
相信聰明的你已經(jīng)知道了冒泡排序是一種依次比較相鄰元素并按照相應(yīng)規(guī)則進(jìn)行排序的排序方法。
在這種排序過程中,重要的元素共有兩個(gè),分別是冒泡排序的趟數(shù)以及比較的次數(shù)。一般而言,趟數(shù)是元素的個(gè)數(shù)減一得到,為什么不等于元素的個(gè)數(shù)呢?因?yàn)槊窟M(jìn)行一趟冒泡排序,就會(huì)有一個(gè)元素來(lái)到它正確的位置,所以當(dāng)除了最后一個(gè)元素以外的其他元素都來(lái)到了正確位置的時(shí)候,那么最后一個(gè)元素也一定處于正確位置。
說(shuō)完趟數(shù),再來(lái)聊聊比較次數(shù)。有親自實(shí)踐的小伙伴們一定發(fā)現(xiàn)了在進(jìn)行第一趟冒泡排序的時(shí)候,比較的次數(shù)是最多的,是除了第一個(gè)元素以外的所有元素都要與之比較,也就是元素的個(gè)數(shù)減一,但是隨著趟數(shù)的增加,比較的次數(shù)也會(huì)隨之減少,究其原因,是因?yàn)槊拷?jīng)過一次冒泡排序,就會(huì)有一個(gè)元素來(lái)到正確的位置,那么這個(gè)正確的元素也就無(wú)需參加后續(xù)的比較了。
那么具體的代碼實(shí)現(xiàn)究竟是怎么樣呢?接下來(lái)就讓我們一起揭開它神秘的面紗,一探究竟!
1.2具體代碼的實(shí)現(xiàn)
void bubble_sort(int arr[], int sz) { //趟數(shù) 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; } } } }
2.qsort函數(shù)
在了解完冒泡排序之后,有的小伙伴還是第一次見到qsort函數(shù),那么下面我們就來(lái)簡(jiǎn)單介紹一下qsort函數(shù)。
首先我們要清楚的是qsort函數(shù)是庫(kù)里面的函數(shù),所以我們?cè)谑褂玫臅r(shí)候要引用頭文件<stdlib.h>。
然后我們?cè)賮?lái)了解一下這個(gè)函數(shù)的各個(gè)部分。第一個(gè)是返回類型,從上圖中我們可以看到返回類型是void,也就是說(shuō)當(dāng)我們?cè)谝眠@個(gè)函數(shù)的時(shí)候,它并不會(huì)返回任何參數(shù)。第二個(gè)是參數(shù)部分,第一個(gè)參數(shù)是一個(gè)指針變量,第二、三個(gè)參數(shù)是無(wú)符號(hào)的整型變量,第四個(gè)就是我們之前講解過的函數(shù)指針變量。在第四個(gè)參數(shù)部分也就體現(xiàn)了回調(diào)函數(shù)這一功能。
最后我們來(lái)講講這個(gè)函數(shù)該怎么使用。在引用完頭文件后,我們下一步就是要對(duì)這個(gè)函數(shù)進(jìn)行傳參,在意義上分別是要排序的對(duì)象,元素個(gè)數(shù),元素大?。ㄒ宰止?jié)為單位),判斷是否交換的函數(shù)(根據(jù)需求自寫)。
說(shuō)到第四個(gè)參數(shù)所指向的函數(shù)需要根據(jù)需求自己來(lái)寫,可能有小伙伴就疑惑了,到底要怎么寫呢。別怕,上面這幅圖片可以說(shuō)是為我們函數(shù)的書寫提供了方向。首先你要確保你書寫的函數(shù)的返回類型為int,并且依據(jù)交換的原則來(lái)進(jìn)行返回值的代碼書寫。拿整型數(shù)組排序?yàn)槔?,如果你想要最終數(shù)組的整形呈現(xiàn)升序排序,那么如果兩個(gè)數(shù)是降序排序就應(yīng)該返回小于0的整型。具體代碼的實(shí)現(xiàn)如下,以整形數(shù)組和結(jié)構(gòu)體為例。
//用qsort函數(shù)實(shí)現(xiàn)各種類型的排序 //整形數(shù)組的排序 int cmp_int(void const* e1, void const* e2) { return *(int*)e1 - *(int*)e2; } int main() { int i = 0; int arr[10] = { 9,8,7,6,5,4,3,2,1,0 }; int sz = sizeof(arr) / sizeof(arr[0]); qsort(arr, sz, 4, cmp_int); for (i = 0; i < 10; i++) { printf("%d ", arr[i]); } return 0; } //結(jié)構(gòu)體的排序 struct stu { char name[20]; int age; }; int cmp_age(void const*e1, void const*e2) { return (((struct stu*)e1)->age - ((struct stu*)e2)->age); } int cmp_name(void const* e1, void const* e2) { return strcmp(((struct stu*)e1)->name ,((struct stu*)e2)->name); } int main() { struct stu Stu[3] = { {"zhangsan", 20},{"wangwu", 42},{"lisi",29}}; int sz = sizeof(Stu) / sizeof(Stu[0]); //qsort(Stu, sz, sizeof(Stu[0]), cmp_age); qsort(Stu, sz, sizeof(Stu[0]), cmp_name); return 0; }
3.qsort函數(shù)的實(shí)現(xiàn)
隨著我們學(xué)習(xí)的不斷深入,類型的不斷豐富,我們會(huì)發(fā)現(xiàn)上面的冒泡排序已然滿足不了我們的需求,諸如結(jié)構(gòu)體的排序,這種代碼已經(jīng)是無(wú)能為力了。那么接下來(lái)我們就自己來(lái)利用冒泡排序的原理來(lái)書寫qsort函數(shù)。
//實(shí)現(xiàn)元素的交換 void swap(char* e1, char* e2, size_t sz) { size_t i = 0; for (i = 0; i < sz; i++) { char tmp = 0; tmp = *e1; *e1 = *e2; *e2 = tmp; e1++; e2++; } } //qsort函數(shù)的自定義 void bubble_sort(void* base, size_t num, size_t size, int (*compar)(const void*e1, const void*e2)) { size_t i = 0; size_t j = 0; //冒泡排序的趟數(shù) for (i = 0; i < num-1; i++) { //比較的次數(shù) for (j = 0; j < num - i - 1; j++) { if (compar((char *)base+j*size,(char*)base+(j+1)*size)>0) { //交換 swap((char*)base + j*size, (char*)base + (j + 1)*size, size); } } } }
到此這篇關(guān)于C語(yǔ)言qsort函數(shù)用冒泡排序?qū)崿F(xiàn)過程詳解的文章就介紹到這了,更多相關(guān)C語(yǔ)言qsort函數(shù)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C/C++中static,const,inline三種關(guān)鍵字詳細(xì)總結(jié)
以下是對(duì)C/C++中static,const,inline的三種關(guān)鍵字進(jìn)行了詳細(xì)的分析介紹,需要的朋友可以過來(lái)參考下2013-09-09gazebo里通過節(jié)點(diǎn)發(fā)布topic讓關(guān)節(jié)轉(zhuǎn)動(dòng)實(shí)現(xiàn)詳解
這篇文章主要介紹了gazebo里通過節(jié)點(diǎn)發(fā)布topic讓關(guān)節(jié)轉(zhuǎn)動(dòng)實(shí)現(xiàn)詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-12-12C++中實(shí)現(xiàn)保存數(shù)據(jù)到CSV文件
這篇文章主要介紹了C++中實(shí)現(xiàn)保存數(shù)據(jù)到CSV文件方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-08-08C語(yǔ)言編程計(jì)算信噪比SNR理解學(xué)習(xí)
這篇文章主要介紹了C語(yǔ)言編程信噪比SNR計(jì)算的理解學(xué)習(xí),信噪比,英文名稱叫做SNR或S/N(SIGNAL-NOISE RATIO)。是指一個(gè)電子設(shè)備或者電子系統(tǒng)中信號(hào)與噪聲的比例2021-10-10詳解C++中構(gòu)造函數(shù),拷貝構(gòu)造函數(shù)和賦值函數(shù)的區(qū)別和實(shí)現(xiàn)
這篇文章主要介紹了C++中構(gòu)造函數(shù),拷貝構(gòu)造函數(shù)和賦值函數(shù)的區(qū)別和實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-03-03C語(yǔ)言實(shí)現(xiàn)簡(jiǎn)易通訊錄實(shí)例
大家好,本篇文章主要講的是C語(yǔ)言實(shí)現(xiàn)簡(jiǎn)易通訊錄實(shí)例,感興趣的同學(xué)趕快來(lái)看一看吧,對(duì)你有幫助的話記得收藏一下2022-02-02C++實(shí)現(xiàn)簡(jiǎn)易萬(wàn)年歷
這篇文章主要為大家詳細(xì)介紹了C++實(shí)現(xiàn)簡(jiǎn)易萬(wàn)年歷,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-10-10