快速排序里的學問:霍爾快排的實現

霍爾快排的C語言實現
服務器君一共花費了157.148 ms進行了5次數據庫查詢,努力地為您提供了這個頁面。
試試閱讀模式?希望聽取您的建議

專題的前一篇講了快速排序的始祖——霍爾快排,那么這里就簡單地實現一下霍爾快排。

補充說明下,快排的一個核心步驟是選取樞紐元,通常的做法是將第一個元素用作樞紐元,《算法導論》里的快排例子和Hoare快排都是這種樞紐元選擇。先撇開效率不說,我們先看看Hoare快排的實現:

#include "stdio.h"
#include "math.h"
#include "stdlib.h"

int num = 10;

void PrintArray(int arr[])
{
    int i;
	for(i=0; i < num; ++i)
	{
	    printf("%d ", arr[i]);
	}
}

//一趟快排之后,以樞紐為分隔,左邊的<=樞紐, 右邊的>=樞紐
int Partition(int *arr, int beg, int end)
{
	int low = beg, high = end;
	//選定樞軸
	int sentinel = arr[beg];
	while(low < high)
	{
	    //printf("\n    定點取arr[%d]的值,設為 sentinel(%d)", low, sentinel );
	    //printf("\n    比sentinel(%d)大的都丟到右邊", sentinel);
	    //比樞紐小的交換到低端
		while(low < high && arr[high]>=sentinel)
		{
            //printf("\n    arr[%d](%d) >= sentinel(%d)", high, arr[high], sentinel);
		    --high;
		    //printf(". high自減為%d, 此時 arr[high] 為 %d", high, arr[high]);
		}
		arr[low] = arr[high];
		//printf("\n    賦值-> arr[low](arr[%d]) = arr[high](arr[%d]) = %d", low, high, arr[low]);
		//printf("\n    比sentinel(%d)小的都丟到左邊", sentinel);
		//比樞紐大的交換到高端
		while(low < high && arr[low]<=sentinel)
		{

		    //printf("\n    arr[%d](%d) <= sentinel(%d)", low, arr[low], sentinel);
		    ++low;
		    //printf(". low自增為%d, 此時 arr[low] 為 %d", low, arr[low]);
		}
		arr[high] = arr[low];
		//printf("\n    賦值-> arr[high](arr[%d]) = arr[low](arr[%d]) = %d", high, low, arr[high]);
	}
	arr[low] = sentinel;

	printf("\n排序過程:");
	PrintArray(arr);
	return low;
}

void QuickSort(int *arr, int beg, int end)
{
	if(beg < end)
	{
		int pivot = Partition(arr, beg, end);
		//分治思想,遞歸排序
		QuickSort(arr, beg, pivot-1);
		QuickSort(arr, pivot+1, end);
	}
}

int main()
{
    int i;
	int arr[10];

	srand(time(0));
    for(i=0; i < 10; i++)
    {
        arr[i] = rand()%100+1;
        //printf("%d ", rand()%100+1);
    }
    printf("初始數組:");
    PrintArray(arr);

	QuickSort(arr, 0, num-1);

	printf("\n最后結果:");
	PrintArray(arr);

	return 0;
}

程序運行結果為:

初始數組:80 16 97 6 12 92 31 52 54 89
排序過程: [ 54 16 52 6 12 31 ] 80 [ 92 97 89 ]
排序過程:[ 31 16 52 6 12 ] 54 [ 80 92 97 89 ]
排序過程:[ 12 16 6 ] 31 [ 52 54 80 92 97 89 ]
排序過程:[ 6 ] 12 [ 16 31 52 54 80 92 97 89 ])
排序過程:[ 6 12 16 31 52 54 80 89 ] 92 [ 97 ]
最后結果:6 12 16 31 52 54 80 89 92 97
Process returned 0 (0x0)   execution time : 0.384 s
Press any key to continue.

排序的思路是,選定一個樞紐元,比樞紐元大的全部丟到右邊,比樞紐元小的全部丟到左邊,可以看看下圖:

霍爾快排的思路清晰了吧?

前面提到了,《算法導論》里的快排例子和Hoare快排都是將第一個元素用作樞紐元的排序,當然也有其它選擇法,后面會介紹到。

延伸閱讀

此文章所在專題列表如下:

  1. 快速排序里的學問:從猜數字開始
  2. 快速排序里的學問:再看看稱球問題
  3. 快速排序里的學問:信息熵
  4. 快速排序里的學問:快速排序的過程
  5. 快速排序里的學問:霍爾與快速排序
  6. 快速排序里的學問:霍爾快排的實現
  7. 快速排序里的學問:樞紐元選擇與算法效率
  8. 快速排序里的學問:隨機化快排

本文地址:http://www.zqhthc.tw/librarys/veda/detail/2396,歡迎訪問原出處。

不打個分嗎?

轉載隨意,但請帶上本文地址:

http://www.zqhthc.tw/librarys/veda/detail/2396

如果你認為這篇文章值得更多人閱讀,歡迎使用下面的分享功能。
小提示:您可以按快捷鍵 Ctrl + D,或點此 加入收藏

大家都在看

閱讀一百本計算機著作吧,少年

很多人覺得自己技術進步很慢,學習效率低,我覺得一個重要原因是看的書少了。多少是多呢?起碼得看3、4、5、6米吧。給個具體的數量,那就100本書吧。很多人知識結構不好而且不系統,因為在特定領域有一個足夠量的知識量+足夠良好的知識結構,系統化以后就足以應對大量未曾遇到過的問題。

奉勸自學者:構建特定領域的知識結構體系的路徑中再也沒有比學習該專業的專業課程更好的了。如果我的知識結構體系足以囊括面試官的大部分甚至吞并他的知識結構體系的話,讀到他言語中的一個詞我們就已經知道他要表達什么,我們可以讓他坐“上位”畢竟他是面試官,但是在知識結構體系以及心理上我們就居高臨下。

所以,閱讀一百本計算機著作吧,少年!

《UNIX環境高級編程(第2版)》 史蒂文斯 (作者), 拉戈 (作者), 尤晉元 (譯者), 張亞英 (譯者), 戚正偉 (譯者)

《UNIX環境高級編程(第2版)》是被譽為UNIX編程“圣經”的Advanced Programming in the UNIX Environment一書的更新版。在本書第1版出版后的十幾年中,UNIX行業已經有了巨大的變化,特別是影響UNIX編程接口的有關標準變化很大。本書在保持了前一版的風格的基礎上,根據最新的標準對內容進行了修訂和增補,反映了最新的技術發展。書中除了介紹UNIX文件和目錄、標準I/O庫、系統數據文件和信息、進程環境、進程控制、進程關系、信號、線程、線程控制、守護進程、各種I/O、進程間通信、網絡IPC、偽終端等方面的內容,還在此基礎上介紹了多個應用示例,包括如何創建數據庫函數庫以及如何與網絡打印機通信等。

更多計算機寶庫...

英超直播吻球网