亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > C++ > 正文

C語言實現排序算法之歸并排序詳解

2020-05-23 14:24:03
字體:
來源:轉載
供稿:網友
這篇文章主要介紹了C語言實現排序算法之歸并排序,對歸并排序的原理及實現過程做了非常詳細的解讀,需要的朋友可以參考下
 
 

排序算法中的歸并排序(Merge Sort)是利用"歸并"技術來進行排序。歸并是指將若干個已排序的子文件合并成一個有序的文件。

一、實現原理:

1、算法基本思路

設兩個有序的子文件(相當于輸入堆)放在同一向量中相鄰的位置上:R[low..m],R[m+1..high],先將它們合并到一個局部的暫存向量R1(相當于輸出堆)中,待合并完成后將R1復制回R[low..high]中。

(1)合并過程

合并過程中,設置i,j和p三個指針,其初值分別指向這三個記錄區的起始位置。合并時依次比較R[i]和R[j]的關鍵字,取關鍵字較小的記錄復制到R1[p]中,然后將被復制記錄的指針i或j加1,以及指向復制位置的指針p加1。
重復這一過程直至兩個輸入的子文件有一個已全部復制完畢(不妨稱其為空),此時將另一非空的子文件中剩余記錄依次復制到R1中即可。
最后,將結果賦值的R[]中。

(2)動態申請R1

實現時,R1是動態申請的,因為申請的空間可能很大,故須加入申請空間是否成功的處理。

二、3種方法實現:

算法1:歸并函數都動態分配一個數組,兩個有序數組合并成一個有序數組

//合并將兩個有序序列([low,mid],[mid+1,high])合并void Merge(int arr[],int low,int mid,int high){  int i=low,j=mid+1,p=0;  int *newarr = (int *)malloc((high-low+1)*sizeof(int));//用來暫存排序好的數據  if(!newarr){    printf("malloc error!/n");    exit(1);  }  while(i<=mid && j<=high){    //以下過程很類似兩個有序字符串合并成一個有序字符串    if(arr[i] < arr[j])      newarr[p++] = arr[i++];    else      newarr[p++] = arr[j++];  }  while(i<=mid)    newarr[p++] = arr[i++];  while(j<=high)    newarr[p++] = arr[j++];  for(i=low,p=0;p<(high-low+1);i++,p++)  //將結果復制到原數組當中    arr[i] = newarr[p];  free(newarr);}

算法2:

程序開始處就動態分配一個大數組,避免每次都要創建很多小數組,釋放內存的時候,不會立即釋放。

有關assert()參見:http://www.49028c.com/article/39685.htm

/* * File: mergesort.c * Time: 2014-07-19 HJJ */#include <stdio.h>#include <stdlib.h>#include <assert.h>static void merge1(int array[], int tmp[], int lpos, int rpos, int rend);static void msort1(int array[], int tmp[], int left, int right);void merge_sort1(int array[], int n){ assert(array!=NULL && n>1); //條件不滿足,退出程序并打印錯誤語句。 int *tmp = (int *)malloc(sizeof(int) * n); assert(tmp != NULL); int i; for (i = 0; i < n; i ++) { tmp[i] = array[i]; } msort1(array, tmp, 0, n-1); free(tmp);}//遞歸的調用此函數,實現折半劃分,只完成劃分,不實現排序,最終返回array[]數組有序static void msort1(int array[], int tmp[], int left, int right){ assert(array!=NULL && tmp!=NULL); if (left == right) return; int center = (left + right) / 2; msort1(tmp, array, left, center); msort1(tmp, array, center+1, right); merge1(tmp, array, left, center+1, right);}//該函數實現,將array[]的左右兩半排好序的數組,歸并為tmp[],并排序static void merge1(int array[], int tmp[], int lpos, int rpos, int rend){ assert(array!=NULL && tmp!=NULL); int lend = rpos - 1; int tmp_pos = lpos; while (lpos<=lend && rpos<=rend) { if (array[lpos] <= array[rpos])  tmp[tmp_pos++] = array[lpos++]; else  tmp[tmp_pos++] = array[rpos++]; } while (lpos <= lend) tmp[tmp_pos++] = array[lpos++]; while (rpos <= rend) tmp[tmp_pos++] = array[rpos++];}int main(int argc, char *argv[]){  int a[7] = {6, 5, 4, 3, 2, 1, 7};  merge_sort1(a, 7);  int i;  for (i = 0; i < 7; i ++) {    printf("%3d", a[i]);  }  printf("/n");  return 0;}

算法3:
程序開始處分配一個大的數組,只是每次用array[]將數據給tmp[]排好序后,最后再將tmp[]給array[]賦值,這樣就能完成每次調用的時候,入口都一樣。

void merge_sort1(int array[], int n){ assert(array!=NULL && n>1); //條件不滿足,退出程序并打印錯誤語句。 int *tmp = (int *)malloc(sizeof(int) * n); assert(tmp != NULL); int i; for (i = 0; i < n; i ++) { tmp[i] = array[i]; } msort1(array, tmp, 0, n-1); free(tmp);}//遞歸的調用此函數,實現折半劃分,只完成劃分,不實現排序,最終返回array[]數組有序static void msort1(int array[], int tmp[], int left, int right){ assert(array!=NULL && tmp!=NULL); if (left == right) return; int center = (left + right) / 2; msort1(tmp, array, left, center); msort1(tmp, array, center+1, right); merge(tmp, array, left, center+1, right);}

實現方法二:

void merge(int array[],int tmp[],int lpos,int rpos,int rend){  int i,leftend,num,tmppos;  leftend = rpos - 1;  num = rend - lpos + 1;  tmppos = lpos;  while(lpos <= leftend && rpos <= rend){    if(array[lpos] <= array[rpos])      tmp[tmppos++] = array[lpos++];    else      tmp[tmppos++] = array[rpos++];  }  while(lpos <= leftend)    tmp[tmppos++] = array[lpos++];  while(rpos <= rend)    tmp[tmppos++] = array[rpos++];  for(i = 0;i < num;i++,rend--)    array[rend] = tmp[rend];}

歸并排序:將一個無序數組合并成一個有序數組

有兩種實現方法:自底向上和自頂向下

1、 自底向上的方法(自底向上的歸并排序算法雖然效率較高,但可讀性較差。)

(1) 自底向上的基本思想:

自底向上的基本思想是:第1趟歸并排序時,將待排序的文件R[1..n]看作是n個長度為1的有序子文件,將這些子文件兩兩歸并,若n為偶數,則得到n/2個長度為2的有序子文件;若n為奇數,則最后一個子文件輪空(不參與歸并)。故本趟歸并完成后,前logn個有序子文件長度為2,但最后一個子文件長度仍為1;第2趟歸并則是將第1趟歸并所得到的logn個有序的子文件兩兩歸并,如此反復,直到最后得到一個長度為n的有序文件為止。
上述的每次歸并操作,均是將兩個有序的子文件合并成一個有序的子文件,故稱其為"二路歸并排序"。
類似地有k(k>2)路歸并排序。

(2) 一趟歸并算法
 分析:
在某趟歸并中,設各子文件長度為length(最后一個子文件的長度可能小于length),則歸并前R[1..n]中共有 個有序的子文件:R[1..length],R[length+1..2length],…

注意:

調用歸并操作將相鄰的一對子文件進行歸并時,必須對子文件的個數可能是奇數、以及最后一個子文件的長度小于length這兩種特殊情況進行特殊處理:

① 若子文件個數為奇數,則最后一個子文件無須和其它子文件歸并(即本趟輪空);
② 若子文件個數為偶數,則要注意最后一對子文件中后一子文件的區間上界是n。

具體算法如下:

/*自底向上,這里就不寫真正的代碼了,從網上copy了*/void MergePass(SeqList R,int length){ //對R[1..n]做一趟歸并排序  int i;  for(i=1;i+2*length-1<=n;i=i+2*length)    Merge(R,i,i+length-1,i+2*length-1);  //歸并長度為length的兩個相鄰子文件  if(i+length-1<n) //尚有兩個子文件,其中后一個長度小于length    Merge(R,i,i+length-1,n); //歸并最后兩個子文件//注意:若i≤n且i+length-1≥n時,則剩余一個子文件輪空,無須歸并} //MergePassvoid MergeSort(SeqList R){//采用自底向上的方法,對R[1..n]進行二路歸并排序  int length;  for(1ength=1;length<n;length*=2) //做 趟歸并    MergePass(R,length); //有序段長度≥n時終止}

2、自頂向下的方法

采用分治法進行自頂向下的算法設計,形式更為簡潔。

(1)分治法的三個步驟

設歸并排序的當前區間是R[low..high],分治法的三個步驟是:

分解:將當前區間一分為二,即求分裂點:mid = (low+high)/2;
求解:遞歸地對兩個子區間R[low..mid]和R[mid+1..high]進行歸并排序;
組合:將已排序的兩個子區間R[low..mid]和R[mid+1..high]歸并為一個有序的區間R[low..high]。

遞歸的終結條件:子區間長度為1(一個記錄自然有序)。

具體算法:

void MSort(int arr[],int low,int high){  if(low < high){    int mid = (low+high)/2;    MSort(arr,low,mid);   //左半區排序    MSort(arr,mid+1,high); //右半區排序    Merge(arr,low,mid,high);//左右半區合并  }}

三:分析

1、穩定性
歸并排序是一種穩定的排序。

2、存儲結構要求
可用順序存儲結構。也易于在鏈表上實現。

3、時間復雜度
對長度為n的文件,需進行lgn趟二路歸并,每趟歸并的時間為O(n),故其時間復雜度無論是在最好情況下還是在最壞情況下均是O(nlgn)。

4、空間復雜度
需要一個輔助向量來暫存兩有序子文件歸并的結果,故其輔助空間復雜度為O(n),顯然它不是就地排序。

注意:
若用單鏈表做存儲結構,很容易給出就地的歸并排序。


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲精品一区在线观看香蕉| 欧美日韩999| 欧美在线一区二区三区四| 国产精品69久久久久| 久久亚洲影音av资源网| 国产精品一区二区女厕厕| 午夜精品福利电影| 亚洲天堂日韩电影| 亚洲激情视频在线播放| 久久久精品国产一区二区| 亚洲第一免费播放区| 精品国产精品自拍| 亚洲女人天堂av| 国模极品一区二区三区| 91久久综合亚洲鲁鲁五月天| 国产精品美女主播在线观看纯欲| 欧美最顶级的aⅴ艳星| 日韩欧美国产黄色| 国产精品一区二区三区在线播放| 亚洲国产欧美一区二区丝袜黑人| 国产成人高潮免费观看精品| 在线观看欧美视频| 欧美成人精品h版在线观看| 富二代精品短视频| 日韩欧美aaa| 69久久夜色精品国产69乱青草| 91国内免费在线视频| 国产精品高潮呻吟久久av黑人| www国产精品com| 亚洲护士老师的毛茸茸最新章节| 日本一区二三区好的精华液| 亚洲高清久久网| 国产一区二区三区在线播放免费观看| 97精品一区二区视频在线观看| 久久久久久国产三级电影| 亚洲午夜av久久乱码| 在线成人中文字幕| 亚洲精品久久久久国产| 亚洲精品国产美女| 海角国产乱辈乱精品视频| 精品久久久久久国产91| 国产精品9999| 欧美性感美女h网站在线观看免费| 亚洲福利视频二区| 久久久久国产精品免费| 中文字幕在线精品| 欧美国产第一页| 国产亚洲精品久久久久久牛牛| 国产精品扒开腿爽爽爽视频| 国产精品久久999| 国外成人免费在线播放| 亚洲人av在线影院| 2019av中文字幕| 欧美激情综合色| 国产自产女人91一区在线观看| 精品久久久在线观看| 欧美性xxxxx极品娇小| 亚洲男人天堂2023| 麻豆国产va免费精品高清在线| 亚洲影视九九影院在线观看| 国产精品xxx视频| 亚洲女性裸体视频| 国产欧美一区二区白浆黑人| 亚洲人成网在线播放| 欧美中文在线观看| 成人免费福利在线| 色琪琪综合男人的天堂aⅴ视频| 亚洲精品国产电影| 日韩黄色高清视频| 久久久久成人网| 精品久久久久久| 日本久久久久亚洲中字幕| 成人黄色免费片| 91av视频在线免费观看| 日本一区二区在线免费播放| 国产精品羞羞答答| 国产亚洲欧洲高清| 国产日产欧美a一级在线| 日韩高清电影免费观看完整版| 精品久久香蕉国产线看观看亚洲| 亚洲一区二区三区成人在线视频精品| 国产欧美日韩专区发布| 伊人久久久久久久久久久| 亚洲aa中文字幕| 国产ts一区二区| 国产精品成人在线| 国产99视频精品免视看7| 日韩一区二区在线视频| 日韩在线观看免费高清完整版| 91av视频在线免费观看| 亚洲xxxx做受欧美| 久久精品视频在线| 久久久亚洲精品视频| 精品久久久在线观看| 国产精品人成电影在线观看| 亚洲精品福利免费在线观看| 国产精品久久久久久五月尺| 亚洲人成亚洲人成在线观看| 国产高清视频一区三区| 日本中文字幕成人| 久久大大胆人体| 日韩欧美成人网| 欧美极品美女视频网站在线观看免费| 国产欧美亚洲视频| 国产精品草莓在线免费观看| 欧美性高潮在线| 久久夜色精品亚洲噜噜国产mv| 在线丨暗呦小u女国产精品| 久久香蕉国产线看观看av| 国产精品久久久久久久天堂| 2018中文字幕一区二区三区| 国内精品久久久久久久| 欧美激情成人在线视频| 92福利视频午夜1000合集在线观看| 久久久久国产视频| 亚洲成人在线视频播放| 欧美日韩美女视频| 国产成人精品午夜| 国产精品美女久久久久av超清| 91在线视频成人| 日韩中文字幕网址| 欧美日韩成人网| 国产一区二区免费| 国产欧美一区二区三区在线| 深夜福利91大全| 欧美精品18videos性欧美| 懂色av中文一区二区三区天美| 91九色在线视频| 亚洲国产成人爱av在线播放| 亚洲一区美女视频在线观看免费| xxx欧美精品| 国产精品入口日韩视频大尺度| 国产午夜精品理论片a级探花| 日韩高清av在线| 大伊人狠狠躁夜夜躁av一区| 精品国产一区二区三区久久狼5月| 日韩欧美国产高清91| 亚洲欧美一区二区精品久久久| 欧美另类在线观看| 欧美极品xxxx| 热re91久久精品国99热蜜臀| 久久噜噜噜精品国产亚洲综合| 日韩在线观看免费av| 亚洲成人黄色网| 亚洲欧美三级在线| 日韩av三级在线观看| 久久99视频免费| 日韩av一卡二卡| 欧美国产日韩一区| 91亚洲人电影| 久久免费在线观看| 一区二区欧美激情| 日本精品中文字幕| 日韩欧美中文字幕在线播放| 在线观看不卡av| 中文字幕亚洲一区二区三区| 亚洲国产中文字幕在线观看| 日韩精品免费综合视频在线播放| 欧美超级免费视 在线| 国产精品免费一区二区三区都可以| 亚洲精选一区二区| 国产日韩精品在线观看| 欧美在线一区二区三区四| **欧美日韩vr在线|