數(shù)據(jù)結構嚴蔚敏版第十章答案
《數(shù)據(jù)結構嚴蔚敏版第十章答案》由會員分享,可在線閱讀,更多相關《數(shù)據(jù)結構嚴蔚敏版第十章答案(15頁珍藏版)》請在裝配圖網(wǎng)上搜索。
1、第十章 內部排序 10.23 void Insert_Sort1(SqList &L)/監(jiān)視哨設在高下標端的插入排序算法k=L.length;for(i=k-1;i;-i) /從后向前逐個插入排序if(L.ri.keyL.ri+1.key)L.rk+1.key=L.ri.key; /監(jiān)視哨for(j=i+1;L.rj.keyL.ri.key;+j)L.rj-1.key=L.rj.key; /前移L.rj-1.key=L.rk+1.key; /插入/Insert_Sort1 10.24 void BiInsert_Sort(SqList &L)/二路插入排序的算法int dMAXSIZE; /輔
2、助存儲x=L.r.key;d=x;first=1;final=1;for(i=2;i=x) /插入前部for(j=final;djL.ri.key;j-)dj+1=dj;dj+1=L.ri.key;final+;else /插入后部for(j=first;djL.ri.key;j+)dj-1=dj;d(j-2)%MAXSIZE+1=L.ri.key;first=(first-2)%MAXSIZE+1; /這種形式的表達式是為了兼顧first=1的情況/forfor(i=first,j=1;di;i=i%MAXSIZE+1,j+)/將序列復制回去L.rj.key=di;/BiInsert_Sor
3、t 10.25 void SLInsert_Sort(SLList &L)/靜態(tài)鏈表的插入排序算法L.r0.key=0;L.r0.next=1;L.r1.next=0; /建初始循環(huán)鏈表for(i=2;i=L.length;i+) /逐個插入p=0;x=L.ri.key;while(L.rL.rp.next.keyx&L.rp.next)p=L.rp.next;q=L.rp.next;L.rp.next=i;L.ri.next=q;/forp=L.r0.next;for(i=1;iL.length;i+) /重排記錄的位置while(pi) p=L.rp.next;q=L.rp.next;if
4、(p!=i)L.rpL.ri;L.ri.next=p;p=q;/for/SLInsert_Sort 10.26 void Bubble_Sort1(int a ,int n)/對包含n個元素的數(shù)組a進行改進的冒泡排序change=n-1; /change指示上一趟冒泡中最后發(fā)生交換的元素while(change)for(c=0,i=0;iai+1)aiai+1;c=i+1; /c指示這一趟冒泡中發(fā)生交換的元素change=c;/while/Bubble_Sort1 10.27 void Bubble_Sort2(int a ,int n)/相鄰兩趟是反方向起泡的冒泡排序算法low=0;high
5、=n-1; /冒泡的上下界change=1;while(lowhigh&change)change=0;for(i=low;iai+1)aiai+1;change=1;high-; /修改上界for(i=high;ilow;i-) /從下向上起泡if(aiai-1)aiai-1;change=1;low+; /修改下界/while/Bubble_Sort2 10.28 void Bubble_Sort3(int a ,int n)/對上一題的算法進行化簡,循環(huán)體中只包含一次冒泡int b 3 ; /b0為冒泡的下界,b 2 為上界,b1無用d=1;b0=0;b 2 =n-1; /d為冒泡方向的
6、標識,1為向上,-1為向下change=1;while(b00) /注意這個交換條件aiai+d;change=1;b1+d-=d; /修改邊界d*=-1; /換個方向/while/Bubble_Sort3 10.29 void OE_Sort(int a ,int n)/奇偶交換排序的算法change=1;while(change) change=0;for(i=1;iai+1)aiai+1;change=1;for(i=0;iai+1)aiai+1;change=1;/while/OE_Sort 分析:本算法的結束條件是連續(xù)兩趟比較無交換發(fā)生10.30 typedef struct int
7、 low; int high; boundary; /子序列的上下界類型 void QSort_NotRecurve(int SQList &L)/快速排序的非遞歸算法low=1;high=L.length;InitStack(S); /S的元素為boundary類型while(low2) /如果當前子序列長度大于3且尚未排好序pivot=Partition(L,low,high); /進行一趟劃分if(high-pivotpivot-low)Push(S,pivot+1,high); /把長的子序列邊界入棧high=pivot-1; /短的子序列留待下次排序elsePush(S,low,pi
8、vot-1);low=pivot+1;/ifelse if(lowhigh&high-low3)/如果當前子序列長度小于3且尚未排好序Easy_Sort(L,low,high); /直接進行比較排序low=high; /當前子序列標志為已排好序else /如果當前子序列已排好序但棧中還有未排序的子序列Pop(S,a); /從棧中取出一個子序列l(wèi)ow=a.low;high=a.high;/while/QSort_NotRecurve int Partition(SQList &L,int low,int high)/一趟劃分的算法,與書上相同L.r0=L.rlow;pivotkey=L.rlow
9、.key;while(lowhigh)while(low=pivotkey)high-;L.rlow=L.rhigh;while(lowhigh&L.rlow.keyL.rhigh.key) L.rlowL.rhigh;else /子序列含有三個元素if(L.rlow.keyL.rlow+1.key) L.rlowL.rlow+1;if(L.rlow+1.keyL.rhigh.key) L.rlow+1L.rhigh;if(L.rlow.keyL.rlow+1.key) L.rlowL.rlow+1;/Easy_Sort 10.31 void Divide(int a ,int n)/把數(shù)組a
10、中所有值為負的記錄調到非負的記錄之前l(fā)ow=0;high=n-1;while(lowhigh)while(low=0) high-; /以0作為虛擬的樞軸記錄alowahigh;while(lowhigh&alow0) low+;alowahigh;/Divide 10.32 typedef enum RED,WHITE,BLUE color; /三種顏色 void Flag_Arrange(color a ,int n)/把由三種顏色組成的序列重排為按照紅,白,藍的順序排列i=0;j=0;k=n-1;while(j=k)switch(aj)case RED:aiaj;i+;j+;break;
11、case WHITE:j+;break;case BLUE:ajak;k-; /這里沒有j+;語句是為了防止交換后aj仍為藍色的情況/Flag_Arrange分析:這個算法中設立了三個指針.其中,j表示當前元素;i以前的元素全部為紅色;k以后的元素全部為藍色.這樣,就可以根據(jù)j的顏色,把其交換到序列的前部或者后部. 10.33 void LinkedList_Select_Sort(LinkedList &L)/單鏈表上的簡單選擇排序算法for(p=L;p-next-next;p=p-next)q=p-next;x=q-data;for(r=q,s=q;r-next;r=r-next) /在q
12、后面尋找元素值最小的結點if(r-next-datanext-data;s=r;if(s!=q) /找到了值比q-data更小的最小結點s-nextp-next=s-next;s-next=q;t=q-next;q-next=p-next-next;p-next-next=t; /交換q和s-next兩個結點/for/LinkedList_Select_Sort 10.34 void Build_Heap(Heap &H,int n)/從低下標到高下標逐個插入建堆的算法for(i=2;iH.rk.key)H.rjH.rk;j=k;/for/Build_Heap 10.35 void TriHe
13、ap_Sort(Heap &H)/利用三叉樹形式的堆進行排序的算法for(i=H.length/3;i0;i-)Heap_Adjust(H,i,H.length);for(i=H.length;i1;i-)H.r1H.ri;Heap_Adjust(H,1,i-1);/TriHeap_Sort void Heap_Adjust(Heap &H,int s,int m)/順序表H中,H.rs+1到H.rm已經(jīng)是堆,把H.rs插入并調整成堆rc=H.rs;for(j=3*s-1;j=m;j=3*j-1)if(jm&H.rj.keyH.rj+1.key) j+;if(jm&H.rj.keyH.rj+1
14、.key) j+;H.rs=H.rj;s=j;H.rs=rc;/Heap_Adjust分析:本算法與課本上的堆排序算法相比,只有兩處改動:1.建初始堆時,i的上限從H.length/3開始(為什么?) 2.調整堆的時候,要從結點的三個孩子結點中選擇最大的那一個,最左邊的孩子的序號的計算公式為j=3*s-1(為什么?) 10.36 void Merge_Sort(int a ,int n)/歸并排序的非遞歸算法for(l=1;ln;l*=2) /l為一趟歸并段的段長for(i=0;(2*i-1)*l(n-1)?(n-1):(start2+l-1);/注意end2可能超出邊界Merge(a,sta
15、rt1,end1,start2,end2); /歸并/Merge_Sort void Merge(int a ,int s1,int e1,int s2,int e2)/將有序子序列as1到ae1和as2到ae2歸并為有序序列as1到ae2int bMAXSIZE; /設立輔助存儲數(shù)組bfor(i=s1,j=s2,k=s1;i=e1&j=e2;k+)if(aiaj) bk=ai+;else bk=aj+;while(i=e1) bk+=ai+;while(j=e2) bk+=aj+; /歸并到b中for(i=s1;i=e2;i+) /復制回去ai=bi;/Merge 10.37 void Li
16、nkedList_Merge_Sort1(LinkedList &L)/鏈表結構上的歸并排序非遞歸算法for(l=1;lnext,e2=p;p-next;p=e2)for(i=1,q=p;inext;i+,q=q-next);e1=q;for(i=1;inext;i+,q=q-next);e2=q; /求出兩個待歸并子序列的尾指針if(e1!=e2) LinkedList_Merge(L,p,e1,e2); /歸并/LinkedList_Merge_Sort1 void LinkedList_Merge(LinkedList &L,LNode *p,LNode *e1,LNode *e2)/對
17、鏈表上的子序列進行歸并,第一個子序列是從p-next到e1,第二個是從e1-next到e2q=p-next;r=e1-next; /q和r為兩個子序列的起始位置while(q!=e1-next&r!=e2-next)if(q-datadata) /選擇關鍵字較小的那個結點接在p的后面p-next=q;p=q;q=q-next;elsep-next=r;p=r;r=r-next;/whilewhile(q!=e1-next) /接上剩余部分p-next=q;p=q;q=q-next;while(r!=e2-next)p-next=r;p=r;r=r-next;/LinkedList_Merge
18、10.38 void LinkedList_Merge_Sort2(LinkedList &L)/初始歸并段為最大有序子序列的歸并排序,采用鏈表存儲結構LNode *endMAXSIZE; /設立一個數(shù)組來存儲各有序子序列的尾指針for(p=L-next-next,i=0;p;p=p-next) /求各有序子序列的尾指針if(!p-next|p-datap-next-data) endi+=p;while(end0-next) /當不止一個子序列時進行兩兩歸并j=0;k=0; /j:當前子序列尾指針存儲位置;k:歸并后的子序列尾指針存儲位置for(p=L-next,e2=p;p-next;p=
19、e2) /兩兩歸并所有子序列e1=endj;e2=endj+1; /確定兩個子序列if(e1-next) LinkedList_Merge(L,p,e1,e2); /歸并endk+=e2; /用新序列的尾指針取代原來的尾指針j+=2; /轉到后面兩個子序列/while/LinkedList_Merge_Sort2 void LinkedList_Merge(LinkedList &L,LNode *p,LNode *e1,LNode *e2)/對鏈表上的子序列進行歸并,第一個子序列是從p-next到e1,第二個是從e1-next到e2q=p-next;r=e1-next;while(q!=e1
20、-next&r!=e2-next)if(q-datadata)p-next=q;p=q;q=q-next;elsep-next=r;p=r;r=r-next;/whilewhile(q!=e1-next)p-next=q;p=q;q=q-next;while(r!=e2-next)p-next=r;p=r;r=r-next;/LinkedList_Merge,與上一題完全相同 10.39 void SL_Merge(int a ,int l1,int l2)/把長度分別為l1,l2且l12(l1+l2)的兩個有序子序列歸并為有序序列start1=0;start2=l1; /分別表示序列1和序列
21、2的剩余未歸并部分的起始位置for(i=0;il1;i+) /插入第i個元素for(j=start2;jl1+l2&ajastart1+i;j+); /尋找插入位置k=j-start2; /k為要向右循環(huán)移動的位數(shù)RSh(a,start1,j-1,k);/將astart1到aj-1之間的子序列循環(huán)右移k位start1+=k+1;start2=j; /修改兩序列尚未歸并部分的起始位置/SL_Merge void RSh(int a ,int start,int end,int k)/將astart到aend之間的子序列循環(huán)右移k位,算法原理參見5.18len=end-start+1;for(i=
22、1;i=k;i+)if(len%i=0&k%i=0) p=i; /求len和k的最大公約數(shù)pfor(i=0;ip;i+) /對p個循環(huán)鏈分別進行右移j=start+i;l=start+(i+k)%len;temp=aj;while(l!=start+i)aj=temp;temp=al;al=aj;j=l;l=start+(j-start+k)%len; /依次向右移astart+i=temp;/for/RSh 10.40 書后給出的解題思路在表述上存在問題,無法理解.比如說,把第一個序列劃分為兩個子序列,使其中的第一個子序列含有s1個記錄,0=s1s,第二個子序列有s個記錄.可是題目中并沒有說
23、明,第一個序列的長度2s.請會做的朋友提供解法. 10.41 void Hash_Sort(int a )/對1000個關鍵字為四位整數(shù)的記錄進行排序int b10000;for(i=0;i1000;i+) /直接按關鍵字散列for(j=ai;bj;j=(j+1)%10000);bj=ai;for(i=0,j=0;i1000;j+) /將散列收回a中if(bj)for(x=bj,k=j;bk;k=(k+1)%10000)if(bk=x)ai+=x;bk=0;/if/Hash_Sort 10.42 typedef struct int gt; /大于該記錄的個數(shù) int lt; /小于該記錄的個
24、數(shù) place; /整個序列中比某個關鍵字大或小的記錄個數(shù) int Get_Mid(int a ,int n)/求一個序列的中值記錄的位置place bMAXSIZE;for(i=0;in;i+) /對每一個元素統(tǒng)計比它大和比它小的元素個數(shù)gt和ltfor(j=0;jai) bi.gt+;else if(ajai) bi.lt+;mid=0;min_dif=abs(b0.gt-b0.lt);for(i=0;in;i+) /找出gt值與lt值最接近的元素,即為中值記錄if(abs(bi.gt-bi.lt)min_dif) mid=i;return mid;/Get_Mid 10.43 void
25、Count_Sort(int a ,int n)/計數(shù)排序算法int cMAXSIZE;for(i=0;in;i+) /對每一個元素for(j=0,count=0;jn;j+) /統(tǒng)計關鍵字比它小的元素個數(shù)if(ajai) count+:ci=count;for(i=0;in;i+) /依次求出關鍵字最小,第二小,.,最大的記錄min=0;for(j=0;jn;j+)if(cjcmin) min=j; /求出最小記錄的下標minaiamin; /與第i個記錄交換cmin=INFINITY; /修改該記錄的c值為無窮大以便下一次選取/Count_Sort 10.44 void Enum_Sort
26、(int a ,int n)/對關鍵字只能取v到w之間任意整數(shù)的序列進行排序int numberw+1,posw+1;for(i=0;in;i+) numberai+; /計數(shù)for(pos0=0,i=1;in;i+)posi=posi-1+numi; /pos數(shù)組可以把關鍵字的值映射為元素在排好的序列中的位置for(i=0;in;i+) /構造有序數(shù)組ccposai+=ai;for(i=0;in;i+)ai=ci;/Enum_Sort分析:本算法參考了第五章三元組稀疏矩陣轉置的算法思想,其中的pos數(shù)組和那里的cpot數(shù)組起的是相類似的作用. 10.45 typedef enum 0,1,2
27、,3,4,5,6,7,8,9 digit; /個位數(shù)類型typedef digit3 num; /3位自然數(shù)類型,假設低位存儲在低下標,高位存儲在高下標 void Enum_Radix_Sort(num a ,int n)/利用計數(shù)實現(xiàn)基數(shù)排序,其中關鍵字為3位自然數(shù),共有n個自然數(shù)int number ,pos ;num cMAXSIZE;for(j=0;j3;j+) /依次對個位,十位和百位排序for(i=0;in;i+) numberaij+; /計數(shù)for(pos0=0,i=1;in;i+)posi=posi-1+numi; /把關鍵字的值映射為元素在排好的序列中的位置for(i=0;
28、in;i+) /構造有序數(shù)組ccposaij+=ai;for(i=0;in;i+)ai=ci;/for/Enum_Radix_Sort分析:計數(shù)排序是一種穩(wěn)定的排序方法.正因為如此,它才能夠被用來實現(xiàn)基數(shù)排序. 10.46 typedef struct int key; int pos; Shadow; /影子序列的記錄類型 void Shadow_Sort(Rectype b ,Rectype &a ,int n)/對元素很大的記錄序列b進行排序,結果放入a中,不移動元素Shadow dMAXSIZE;for(i=0;i1&change;i-) /對影子序列執(zhí)行冒泡排序change=0;for(j=0;jdj+1.key)djdj+1;change=1;/forfor(i=0;in;i+) /按照影子序列里記錄的原來位置復制原序列ai=bdi.pos;/Shadow_Sort
- 溫馨提示:
1: 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
2: 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權益歸上傳用戶所有。
3.本站RAR壓縮包中若帶圖紙,網(wǎng)頁內容里面會有圖紙預覽,若沒有圖紙預覽就沒有圖紙。
4. 未經(jīng)權益所有人同意不得將文件中的內容挪作商業(yè)或盈利用途。
5. 裝配圖網(wǎng)僅提供信息存儲空間,僅對用戶上傳內容的表現(xiàn)方式做保護處理,對用戶上傳分享的文檔內容本身不做任何修改或編輯,并不能對任何下載內容負責。
6. 下載文件中如有侵權或不適當內容,請與我們聯(lián)系,我們立即糾正。
7. 本站不保證下載資源的準確性、安全性和完整性, 同時也不承擔用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。
最新文檔
- 微生物的生長繁殖及其控制筆記
- 股票市場投資分析現(xiàn)代金融投資統(tǒng)計分析李臘生課件
- 物料提升機基礎知識專題培訓課件
- 中考生物總復習 第2講 了解生物圈課件 (12)
- 中考生物 第1部分 第二單元 第一章 細胞是生命活動的基本單位復習課件 (8)
- 一級數(shù)學下冊 阿福的新衣教學建議課件 青島五制
- 一級數(shù)學下冊 第一單元 加與減(一)第1課時 買鉛筆習題課件 北師大
- 財務管理工具概述
- 財務成本管理課件
- 多種二尖瓣成形技術治療二尖瓣前葉關閉不全
- 多用電表的原理和使用說課
- 《藝術品》酒泉四中陳軍興
- 國有獨資公司例題
- 淘寶售后模板
- 環(huán)境心理學觀音山公園調研報告_2