數(shù)據(jù)結(jié)構(gòu)實驗報告-答案_第1頁
數(shù)據(jù)結(jié)構(gòu)實驗報告-答案_第2頁
數(shù)據(jù)結(jié)構(gòu)實驗報告-答案_第3頁
數(shù)據(jù)結(jié)構(gòu)實驗報告-答案_第4頁
數(shù)據(jù)結(jié)構(gòu)實驗報告-答案_第5頁
已閱讀5頁,還剩44頁未讀, 繼續(xù)免費閱讀

下載本文檔

版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請進行舉報或認領(lǐng)

文檔簡介

數(shù)據(jù)結(jié)構(gòu) (C語言版)實驗報告專業(yè) 班級 學(xué)號 姓名實驗 1實驗題目:單鏈表的插入和刪除實驗?zāi)康模毫私夂驼莆站€性表的邏輯結(jié)構(gòu)和鏈?zhǔn)酱鎯Y(jié)構(gòu), 掌握單鏈表的基本算法及相關(guān)的時間性能分析。實驗要求:建立一個數(shù)據(jù)域定義為字符串的單鏈表, 在鏈表中不允許有重復(fù)的字符串; 根據(jù)輸入的字符串,先找到相應(yīng)的結(jié)點,后刪除之。實驗主要步驟:1、分析、理解給出的示例程序。2、調(diào)試程序,并設(shè)計輸入數(shù)據(jù)(如:bat,cat,eat,fat,hat,jat,lat,mat,#),測試程序的如下功能:不允許重復(fù)字符串的插入;根據(jù)輸入的字符串,找到相應(yīng)的結(jié)點并刪除。3、修改程序:1)增加插入結(jié)點的功能。2)將建立鏈表的方法改為頭插入法。程序代碼:#include"stdio.h"#include"string.h"#include"stdlib.h"#include"ctype.h"typedefstructnode{

//定義結(jié)點chardata[10];

//結(jié)點的數(shù)據(jù)域為字符串structnode*next;

//結(jié)點的指針域}ListNode;typedefListNode*LinkList;

// 自定義

LinkList

單鏈表類型LinkListCreatListR1();

//函數(shù),用尾插入法建立帶頭結(jié)點的單鏈表LinkListCreatList(void);

//函數(shù),用頭插入法建立帶頭結(jié)點的單鏈表ListNode*LocateNode();

//函數(shù),按值查找結(jié)點voidDeleteList();

//函數(shù),刪除指定值的結(jié)點voidprintlist();

//函數(shù),打印鏈表中的所有值voidDeleteAll();

//函數(shù),刪除所有結(jié)點,釋放內(nèi)存ListNode*AddNode(); //修改程序:增加節(jié)點。用頭插法,返回頭指針//========== 主函數(shù)==============voidmain(){charch[10],num[5];LinkListhead;head=CreatList(); //用頭插入法建立單鏈表,返回頭指針printlist(head); //遍歷鏈表輸出其值printf("Deletenode(y/n):"); //輸入"y"或"n"去選擇是否刪除結(jié)點scanf("%s",num);if(strcmp(num,"y")==0||strcmp(num,"Y")==0){printf("PleaseinputDelete_data:");scanf("%s",ch); //輸入要刪除的字符串DeleteList(head,ch);printlist(head);}printf("Addnode?(y/n):");//輸入"y"或"n"去選擇是否增加結(jié)點scanf("%s",num);if(strcmp(num,"y")==0||strcmp(num,"Y")==0){head=AddNode(head);}printlist(head);DeleteAll(head); //刪除所有結(jié)點,釋放內(nèi)存}//========== 用尾插入法建立帶頭結(jié)點的單鏈表 ===========LinkListCreatListR1(void){charch[10];LinkListhead=(LinkList)malloc(sizeof(ListNode));// 生成頭結(jié)點ListNode*s,*r,*pp;r=head;r->next=NULL;printf("Input#toend "); //輸入"#"代表輸入結(jié)束printf("\nPleaseinputNode_data:");scanf("%s",ch); //輸入各結(jié)點的字符串while(strcmp(ch,"#")!=0){pp=LocateNode(head,ch); //按值查找結(jié)點,返回結(jié)點指針if(pp==NULL){ //沒有重復(fù)的字符串,插入到鏈表中s=(ListNode*)malloc(sizeof(ListNode));strcpy(s->data,ch);r->next=s;r=s;r->next=NULL;}printf("Input#toend ");printf("PleaseinputNode_data:");scanf("%s",ch);}returnhead; //返回頭指針}//========== 用頭插入法建立帶頭結(jié)點的單鏈表 ===========LinkListCreatList(void){charch[100];LinkListhead,p;head=(LinkList)malloc(sizeof(ListNode));head->next=NULL;while(1){printf("Input#toend ");printf("PleaseinputNode_data:");scanf("%s",ch);if(strcmp(ch,"#")){if(LocateNode(head,ch)==NULL){strcpy(head->data,ch);p=(LinkList)malloc(sizeof(ListNode));p->next=head;head=p;}}elsebreak;}returnhead;}//========== 按值查找結(jié)點,找到則返回該結(jié)點的位置,否則返回 NULL==========ListNode*LocateNode(LinkListhead,char*key){ListNode*p=head->next;// 從開始結(jié)點比較while(p!=NULL&&strcmp(p->data,key)!=0)p=p->next; //掃描下一個結(jié)點returnp; //若p=NULL 則查找失敗,否則

//直到 p為NULLp指向找到的值為

或p->datakey的結(jié)點

key

止}//========== 修改程序:增加節(jié)點 =======ListNode*AddNode(LinkListhead){charch[10];ListNode*s,*pp;printf("\nPleaseinputaNewNode_data:");scanf("%s",ch); //輸入各結(jié)點的字符串pp=LocateNode(head,ch);printf("ok2\n");

//按值查找結(jié)點,返回結(jié)點指針if(pp==NULL){ //沒有重復(fù)的字符串,插入到鏈表中s=(ListNode*)malloc(sizeof(ListNode));strcpy(s->data,ch);printf("ok3\n");s->next=head->next;head->next=s;}returnhead;}//==========刪除帶頭結(jié)點的單鏈表中的指定結(jié)點voidDeleteList(LinkListhead,char*key){

=======ListNode*p,*r,*q=head;p=LocateNode(head,key);

//按

key

值查找結(jié)點的if(p==NULL){

//若沒有找到結(jié)點,退出printf("positionerror");exit(0);}while(q->next!=p)

//p

為要刪除的結(jié)點,

q為

p的前結(jié)點q=q->next;r=q->next;q->next=r->next;free(r);

//釋放結(jié)點}//=========== 打印鏈表voidprintlist(LinkListhead){

=======ListNode*p=head->next;while(p){printf("%s, ",p->data);p=p->next;

//從開始結(jié)點打印}printf("\n");}//========== 刪除所有結(jié)點,釋放空間

===========voidDeleteAll(LinkListhead){ListNode*p=head,*r;while(p->next){r=p->next;free(p);p=r;}free(p);}實驗結(jié)果:Input#toendPleaseinputNode_data:batInput#toendPleaseinputNode_data:catInput#toendPleaseinputNode_data:eatInput#toendPleaseinputNode_data:fatInput#toendPleaseinputNode_data:hatInput#toendPleaseinputNode_data:jatInput#toendPleaseinputNode_data:latInput#toendPleaseinputNode_data:matInput#toendPleaseinputNode_data:#mat, lat, jat, hat, fat, eat, cat, bat,Deletenode(y/n):yPleaseinputDelete_data:hatmat, lat, jat, fat, eat, cat, bat,Insertnode(y/n):yPleaseinputInsert_data:putposition:5mat, lat, jat, fat, eat, put, cat, bat,請按任意鍵繼續(xù) ...示意圖:headmat lat jat hat fat eat cat batNULLheadmat lat jat fat eat cat batNULLhatheadmat lat jat fat eat cat batNULLput心得體會:本次實驗使我們對鏈表的實質(zhì)了解更加明確了, 對鏈表的一些基本操作也更加熟練了。 另外實驗指導(dǎo)書上給出的代碼是有一些問題的, 這使我們認識到實驗過程中不能想當(dāng)然的直接編譯執(zhí)行,應(yīng)當(dāng)在閱讀并完全理解代碼的基礎(chǔ)上再執(zhí)行,這才是實驗的意義所在。實驗 2實驗題目:二叉樹操作設(shè)計和實現(xiàn)實驗?zāi)康模赫莆斩鏄涞亩x、性質(zhì)及存儲方式,各種遍歷算法。實驗要求:采用二叉樹鏈表作為存儲結(jié)構(gòu), 完成二叉樹的建立, 先序、中序和后序以及按層次遍歷的操作,求所有葉子及結(jié)點總數(shù)的操作。實驗主要步驟:1、分析、理解程序。2、調(diào)試程序, 設(shè)計一棵二叉樹, 輸入完全二叉樹的先序序列, 用#代表虛結(jié)點(空指針),如ABD###CE##F##,建立二叉樹,求出先序、中序和后序以及按層次遍歷序列,求所有葉子及結(jié)點總數(shù)。實驗代碼#include"stdio.h"#include"stdlib.h"#include"string.h"#defineMax20 //結(jié)點的最大個數(shù)typedefstructnode{chardata;structnode*lchild,*rchild;}BinTNode; //自定義二叉樹的結(jié)點類型typedefBinTNode*BinTree; //定義二叉樹的指針intNodeNum,leaf; //NodeNum 為結(jié)點數(shù), leaf為葉子數(shù)//========== 基于先序遍歷算法創(chuàng)建二叉樹 ==============//=====要求輸入先序序列,其中加入虛結(jié)點

"#"

以示空指針的位置

==========BinTreeCreatBinTree(void){BinTreeT;charch;if((ch=getchar())=='#')return(NULL); //讀入#,返回空指針else{T=(BinTNode*)malloc(sizeof(BinTNode));// 生成結(jié)點T->data=ch;T->lchild=CreatBinTree(); //構(gòu)造左子樹T->rchild=CreatBinTree(); //構(gòu)造右子樹return(T);}}//========NLR 先序遍歷 =============voidPreorder(BinTreeT){if(T){printf("%c",T->data);

//訪問結(jié)點Preorder(T->lchild);

//先序遍歷左子樹Preorder(T->rchild);

//先序遍歷右子樹}}//========LNR 中序遍歷 ===============voidInorder(BinTreeT){if(T){Inorder(T->lchild);printf("%c",T->data);

//中序遍歷左子樹//訪問結(jié)點Inorder(T->rchild);

//中序遍歷右子樹}}//==========LRN 后序遍歷 ============voidPostorder(BinTreeT){if(T){Postorder(T->lchild);

//后序遍歷左子樹Postorder(T->rchild);

//后序遍歷右子樹printf("%c",T->data);

//訪問結(jié)點}}//=====采用后序遍歷求二叉樹的深度、結(jié)點數(shù)及葉子數(shù)的遞歸算法 ========intTreeDepth(BinTreeT){inthl,hr,max;if(T){hl=TreeDepth(T->lchild);

//求左深度hr=TreeDepth(T->rchild);

//求右深度max=hl>hr?hl:hr;

//取左右深度的最大值NodeNum=NodeNum+1;

//求結(jié)點數(shù)if(hl==0&&hr==0)leaf=leaf+1;

//若左右深度為

0,即為葉子。return(max+1);}elsereturn(0);}//====利用"先進先出 "(FIFO

)隊列,按層次遍歷二叉樹

==========voidLevelorder(BinTreeT){intfront=0,rear=1;BinTNode*cq[Max],*p; //定義結(jié)點的指針數(shù)組 cqcq[1]=T; //根入隊while(front!=rear){front=(front+1)%NodeNum;p=cq[front];printf("%c",p->data);

//出隊//出隊,輸出結(jié)點的值if(p->lchild!=NULL){rear=(rear+1)%NodeNum;cq[rear]=p->lchild;

//左子樹入隊}if(p->rchild!=NULL){rear=(rear+1)%NodeNum;cq[rear]=p->rchild;

//右子樹入隊}}}//====數(shù)葉子節(jié)點個數(shù) ==========intcountleaf(BinTreeT){inthl,hr;if(T){hl=countleaf(T->lchild);hr=countleaf(T->rchild);if(hl==0&&hr==0) //若左右深度為 0,即為葉子。return(1);elsereturnhl+hr;}elsereturn0;}//========== 主函數(shù)=================voidmain(){BinTreeroot;chari;intdepth;printf("\n");printf("CreatBin_Tree ; Inputpreorder:");// 輸入完全二叉樹的先序序列,// 用

#代表虛結(jié)點,如

ABD###CE##F##root=CreatBinTree();

//創(chuàng)建二叉樹,返回根結(jié)點do{

//從菜單中選擇遍歷方式,輸入序號。printf("\t**********select************\n");printf("\t1:PreorderTraversal\n");printf("\t2:IorderTraversal\n");printf("\t3:Postordertraversal\n");printf("\t4:PostTreeDepth,Nodenumber,Leafnumber\n");printf("\t5:LevelDepth\n");// 按層次遍歷之前,先選擇 4,求出該樹的結(jié)點數(shù)。printf("\t0:Exit\n");printf("\t*******************************\n");fflush(stdin);scanf("%c",&i); //輸入菜單序號( 0-5)switch(i-'0'){case1:printf("PrintBin_treePreorder:");Preorder(root); //先序遍歷break;case2:printf("PrintBin_TreeInorder:");Inorder(root); //中序遍歷break;case3:printf("PrintBin_TreePostorder:");Postorder(root);

//后序遍歷break;case4:depth=TreeDepth(root); //求樹的深度及葉子數(shù)printf("BinTreeDepth=%d BinTreeNodenumber=%d",depth,NodeNum);printf(" BinTreeLeafnumber=%d",countleaf(root));break;case5:printf("LevePrintBin_Tree:");Levelorder(root); //按層次遍歷break;default:exit(1);}printf("\n");}while(i!=0);}實驗結(jié)果:CreatBin_Tree ; Inputpreorder:ABD###CE##F##**********select************PreorderTraversalIorderTraversalPostordertraversalPostTreeDepth,Nodenumber,LeafnumberLevelDepthExit*******************************1PrintBin_treePreorder:ABDCEF2PrintBin_TreeInorder:DBAECF3PrintBin_TreePostorder:DBEFCA4BinTreeDepth=3BinTreeNodenumber=6BinTreeLeafnumber=35LevePrintBin_Tree:ABCDEF0Pressanykeytocontinue二叉樹示意圖AB CD E F心得體會:這次實驗加深了我對二叉樹的印象, 尤其是對二叉樹的各種遍歷操作有了一定的了解。 同時認識到,在設(shè)計程序時輔以圖形化的描述是非常有用處的。實驗 3實驗題目:圖的遍歷操作實驗?zāi)康模赫莆沼邢驁D和無向圖的概念; 掌握鄰接矩陣和鄰接鏈表建立圖的存儲結(jié)構(gòu); 掌握 DFS及BFS對圖的遍歷操作;了解圖結(jié)構(gòu)在人工智能、工程等領(lǐng)域的廣泛應(yīng)用。實驗要求:采用鄰接矩陣和鄰接鏈表作為圖的存儲結(jié)構(gòu),完成有向圖和無向圖的 DFS和BFS操作。實驗主要步驟:設(shè)計一個有向圖和一個無向圖,任選一種存儲結(jié)構(gòu),完成有向圖和無向圖的 DFS(深度優(yōu)先遍歷)和 BFS(廣度優(yōu)先遍歷)的操作。1. 鄰接矩陣作為存儲結(jié)構(gòu)#include"stdio.h"#include"stdlib.h"#defineMaxVertexNum100 // 定義最大頂點數(shù)typedefstruct{charvexs[MaxVertexNum]; // 頂點表intedges[MaxVertexNum][MaxVertexNum]; //intn,e; // 圖中的頂點數(shù) n和邊數(shù)}MGraph; // 用鄰接矩陣表示的圖的類型//========= 建立鄰接矩陣 =======

e

鄰接矩陣,可看作邊表voidCreatMGraph(MGraph*G){inti,j,k;chara;printf("InputVertexNum(n)andEdgesNum(e):");scanf("%d,%d",&G->n,&G->e);//輸入頂點數(shù)和邊數(shù)scanf("%c",&a);printf("InputVertexstring:");for(i=0;i<G->n;i++){scanf("%c",&a);G->vexs[i]=a;

//

讀入頂點信息,建立頂點表}for(i=0;i<G->n;i++)for(j=0;j<G->n;j++)G->edges[i][j]=0; // 初始化鄰接矩陣printf("Inputedges,CreatAdjacencyMatrix\n");for(k=0;k<G->e;k++){ // 讀入 e條邊,建立鄰接矩陣scanf("%d%d",&i,&j); // 輸入邊( Vi,Vj)的頂點序號G->edges[i][j]=1;G->edges[j][i]=1;// 若為無向圖,矩陣為對稱矩陣;若建立有向圖,去掉該條語句}}//========= 定義標(biāo)志向量,為全局變量typedefenum{FALSE,TRUE}Boolean;Booleanvisited[MaxVertexNum];//========DFS :深度優(yōu)先遍歷的遞歸算法voidDFSM(MGraph*G,inti){// 以Vi為出發(fā)點對鄰接矩陣表示的圖intj;printf("%c",G->vexs[i]); //visited[i]=TRUE; //for(j=0;j<G->n;j++) //if(G->edges[i][j]==1&&!visited[j])DFSM(G,j); //

=============G進行DFS搜索,鄰接矩陣是 0,1訪問頂點 Vi置已訪問標(biāo)志依次搜索 Vi的鄰接點(Vi,Vj)∈E,且 Vj未訪問過,故

矩陣Vj為新出發(fā)點}voidDFS(MGraph*G){inti;for(i=0;i<G->n;i++)visited[i]=FALSE;

//

標(biāo)志向量初始化for(i=0;i<G->n;i++)if(!visited[i])DFSM(G,i);

//Vi//

未訪問過Vi為源點開始

DFS搜索}//===========BFS :廣度優(yōu)先遍歷 =======voidBFS(MGraph*G,intk){ // 以Vk為源點對用鄰接矩陣表示的圖 G進行廣度優(yōu)先搜索inti,j,f=0,r=0;intcq[MaxVertexNum]; // 定義隊列for(i=0;i<G->n;i++)visited[i]=FALSE; // 標(biāo)志向量初始化for(i=0;i<G->n;i++)cq[i]=-1; // 隊列初始化printf("%c",G->vexs[k]); // 訪問源點 Vkvisited[k]=TRUE;cq[r]=k; //Vk 已訪問,將其入隊。注意,實際上是將其序號入隊while(cq[f]!=-1){ // 隊非空則執(zhí)行i=cq[f];f=f+1; //Vf 出隊for(j=0;j<G->n;j++) // 依次Vi的鄰接點 Vjif(G->edges[i][j]==1&&!visited[j]){//Vj 未訪問printf("%c",G->vexs[j]); // 訪問 Vjvisited[j]=TRUE;r=r+1;cq[r]=j; // 訪問過 Vj入隊}}}//==========main=====voidmain(){inti;MGraph*G;G=(MGraph*)malloc(sizeof(MGraph)); // 為圖 G申請內(nèi)存空間CreatMGraph(G); // 建立鄰接矩陣printf("PrintGraphDFS:");DFS(G); // 深度優(yōu)先遍歷printf("\n");printf("PrintGraphBFS:");BFS(G,3); // 以序號為 3的頂點開始廣度優(yōu)先遍歷printf("\n");}2. 鄰接鏈表作為存儲結(jié)構(gòu)#include"stdio.h"#include"stdlib.h"#defineMaxVertexNum50 // 定義最大頂點數(shù)typedefstructnode{ // 邊表結(jié)點intadjvex; // 鄰接點域structnode*next; // 鏈域}EdgeNode;typedefstructvnode{ // 頂點表結(jié)點charvertex; // 頂點域EdgeNode*firstedge; // 邊表頭指針}VertexNode;typedefVertexNodeAdjList[MaxVertexNum]; //AdjListtypedefstruct{AdjListadjlist; // 鄰接表intn,e; // 圖中當(dāng)前頂點數(shù)和邊數(shù)}ALGraph; // 圖類型

是鄰接表類型//========= 建立圖的鄰接表 =======voidCreatALGraph(ALGraph*G){inti,j,k;chara;EdgeNode*s; // 定義邊表結(jié)點printf("InputVertexNum(n)andEdgesNum(e):");scanf("%d,%d",&G->n,&G->e); //

讀入頂點數(shù)和邊數(shù)scanf("%c",&a);printf("InputVertexstring:");for(i=0;i<G->n;i++) //{

建立邊表scanf("%c",&a);G->adjlist[i].vertex=a; //G->adjlist[i].firstedge=NULL;//

讀入頂點信息邊表置為空表}printf("Inputedges,CreatAdjacencyList\n");for(k=0;k<G->e;k++){ //

建立邊表scanf("%d%d",&i,&j); //

讀入邊(

Vi

Vj

)的頂點對序號s=(EdgeNode*)malloc(sizeof(EdgeNode));

//

生成邊表結(jié)點s->adjvex=j; //

鄰接點序號為

js->next=G->adjlist[i].firstedge;G->adjlist[i].firstedge=s; //

將新結(jié)點

*S

插入頂點

Vi

的邊表頭部s=(EdgeNode*)malloc(sizeof(EdgeNode));s->adjvex=i; //

鄰接點序號為

is->next=G->adjlist[j].firstedge;G->adjlist[j].firstedge=s; //

將新結(jié)點

*S

插入頂點

Vj

的邊表頭部}}//========= 定義標(biāo)志向量,為全局變量typedefenum{FALSE,TRUE}Boolean;Booleanvisited[MaxVertexNum];//========DFS :深度優(yōu)先遍歷的遞歸算法voidDFSM(ALGraph*G,inti){ // 以ViEdgeNode*p;printf("%c",G->adjlist[i].vertex); //visited[i]=TRUE; //p=G->adjlist[i].firstedge; //while(p){ //if(!visited[p->adjvex]) //DFSM(G,p->adjvex); //p=p->next; //

=============為出發(fā)點對鄰接鏈表表示的圖 G進行 DFS搜索訪問頂點 Vi標(biāo)記 Vi已訪問取Vi邊表的頭指針依次搜索 Vi的鄰接點 Vj,這里 j=p->adjvex若Vj尚未被訪問則以 Vj為出發(fā)點向縱深搜索找Vi的下一個鄰接點}}voidDFS(ALGraph*G){inti;for(i=0;i<G->n;i++)visited[i]=FALSE;

//

標(biāo)志向量初始化for(i=0;i<G->n;i++)if(!visited[i])

//Vi

未訪問過DFSM(G,i);

//

Vi

為源點開始

DFS搜索}//==========BFS :廣度優(yōu)先遍歷

=========voidBFS(ALGraph*G,intk){ //

Vk

為源點對用鄰接鏈表表示的圖

G進行廣度優(yōu)先搜索inti,f=0,r=0;EdgeNode*p;intcq[MaxVertexNum]; // 定義 FIFO隊列for(i=0;i<G->n;i++)visited[i]=FALSE; // 標(biāo)志向量初始化for(i=0;i<=G->n;i++)cq[i]=-1; // 初始化標(biāo)志向量printf("%c",G->adjlist[k].vertex);// 訪問源點 Vkvisited[k]=TRUE;cq[r]=k; //Vk 已訪問,將其入隊。注意,實際上是將其序號入隊while(cq[f]!=-1){ 隊列非空則執(zhí)行i=cq[f];f=f+1; //Vi 出隊p=G->adjlist[i].firstedge; // 取Vi的邊表頭指針while(p){ // 依次搜索 Vi的鄰接點 Vj(令 p->adjvex=jif(!visited[p->adjvex]){ // 若Vj未訪問過printf("%c",G->adjlist[p->adjvex].vertex); // 訪問 Vjvisited[p->adjvex]=TRUE;r=r+1;cq[r]=p->adjvex; // 訪問過的 Vj入隊

)}p=p->next;

//

Vi

的下一個鄰接點}}//endwhile}//========== 主函數(shù) ===========voidmain(){inti;ALGraph*G;G=(ALGraph*)malloc(sizeof(ALGraph));CreatALGraph(G);printf("PrintGraphDFS:");DFS(G);printf("\n");printf("PrintGraphBFS:");BFS(G,3);printf("\n");}實驗結(jié)果:1.鄰接矩陣作為存儲結(jié)構(gòu)執(zhí)行順序:InputVertexNum(n)andEdgesNum(e):8,9V0InputVertexstring:01234567Inputedges,CreatAdjacencyMatrixV1V201V302V4V5V613142526V7374756PrintGraphDFS:01374256PrintGraphBFS:317042562.鄰接鏈表作為存儲結(jié)構(gòu)執(zhí)行順序:InputVertexNum(n)andEdgesNum(e):8,9InputVertexstring:01234567Inputedges,CreatAdjacencyList01V002V1V21314V325V4V5V6263747V756PrintGraphDFS:02651473PrintGraphBFS:37140265心得體會:這次實驗較以前的實驗難度加大, 必須先理解深度優(yōu)先和廣度優(yōu)先兩種遍歷思路, 和數(shù)據(jù)結(jié)構(gòu)中隊列的基本操作,才能看懂理解代碼。同時圖這種數(shù)據(jù)結(jié)構(gòu)對抽象的能力要求非常高,代碼不容易看懂,排錯也比較麻煩,應(yīng)該多加練習(xí),才能掌握。實驗 4實驗題目:排序?qū)嶒災(zāi)康模赫莆崭鞣N排序方法的基本思想、排序過程、算法實現(xiàn),能進行時間和空間性能的分析,根據(jù)實際問題的特點和要求選擇合適的排序方法。實驗要求:實現(xiàn)直接排序、 冒泡、直接選擇、 快速、堆、歸并排序算法。 比較各種算法的運行速度。實驗主要步驟:實驗代碼#include"stdio.h"#include"stdlib.h"#defineMax100 // 假設(shè)文件長度typedefstruct{ // 定義記錄類型intkey; // 關(guān)鍵字項}RecType;typedefRecTypeSeqList[Max+1];//SeqList 為順序表,表中第 0個元素作為哨兵intn; // 順序表實際的長度//========== 直接插入排序法 ======voidInsertSort(SeqListR){ // 對順序表 R中的記錄 R[1‥n]按遞增序進行插入排序inti,j;for(i=2;i<=n;i++) // 依次插入 R[2],??, R[n]if(R[i].key<R[i-1].key){// 若R[i].key 大于等于有序區(qū)中所有的 keys,則R[i]留在原位R[0]=R[i];j=i-1; //R[0] 是R[i] 的副本do{ // 從右向左在有序區(qū) R[1‥i-1] 中查找 R[i] 的位置R[j+1]=R[j]; // 將關(guān)鍵字大于 R[i].key 的記錄后移j--;}while(R[0].key<R[j].key); // 當(dāng)R[i].key ≥R[j].key 是終止R[j+1]=R[0]; //R[i] 插入到正確的位置上}//endif}//========== 冒泡排序 =======typedefenum{FALSE,TRUE}Boolean;//FALSE 為0,TRUE為1voidBubbleSort(SeqListR){ // 自下向上掃描對 R做冒泡排序inti,j; Booleanexchange; // 交換標(biāo)志for(i=1;i<n;i++){ // 最多做 n-1趟排序exchange=FALSE; // 本趟排序開始前,交換標(biāo)志應(yīng)為假for(j=n-1;j>=i;j--) // 對當(dāng)前無序區(qū) R[i‥n] 自下向上掃描if(R[j+1].key<R[j].key){ // 兩兩比較,滿足條件交換記錄R[0]=R[j+1]; //R[0] 不是哨兵,僅做暫存單元R[j+1]=R[j];R[j]=R[0];exchange=TRUE; // 發(fā)生了交換,故將交換標(biāo)志置為真}if(!exchange)

//

本趟排序未發(fā)生交換,提前終止算法return;}//endfor

(為循環(huán))}//1.======== 一次劃分函數(shù) =====intPartition(SeqListR,inti,intj){//對R[i‥j]做一次劃分,并返回基準(zhǔn)記錄的位置RecTypepivot=R[i]; // 用第一個記錄作為基準(zhǔn)while(i<j){ // 從區(qū)間兩端交替向中間掃描,直到while(i<j&&R[j].key>=pivot.key) // 基準(zhǔn)記錄 pivotj--; // 從右向左掃描,查找第一個關(guān)鍵字小于if(i<j) // 若找到的 R[j].key<pivot.key ,則

i=j相當(dāng)與在位置 ipivot.key 的記錄

上R[j]R[i++]=R[j];//

交換

R[i]

R[j]

,交換后

i

指針加

1while(i<j&&R[i].key<=pivot.key)

//

基準(zhǔn)記錄

pivot

相當(dāng)與在位置

j上i++;

//

從左向右掃描,查找第一個關(guān)鍵字小于

pivot.key

的記錄R[i]if(i<j)

//

若找到的

R[i].key>pivot.key

,則R[j--]=R[i];//

交換

R[i]

R[j]

,交換后

j指針減

1}R[i]=pivot;

//

此時,

i=j

,基準(zhǔn)記錄已被最后定位returni;

//

返回基準(zhǔn)記錄的位置}//2.===== 快速排序 ===========voidQuickSort(SeqListR,intlow,inthigh){ //R[low..high]

快速排序intpivotpos; //

劃分后基準(zhǔn)記錄的位置if(low<high){//pivotpos=Partition(R,low,high);//

僅當(dāng)區(qū)間長度大于 1時才排序?qū)[low..high] 做一次劃分,

得到基準(zhǔn)記錄的位置QuickSort(R,low,pivotpos-1); // 對左區(qū)間遞歸排序QuickSort(R,pivotpos+1,high); // 對右區(qū)間遞歸排序}}//======直接選擇排序voidSelectSort(SeqListR){

========inti,j,k;for(i=1;i<n;i++){

//

做第

i

趟排序(

1≤i

≤n-1

)k=i;for(j=i+1;j<=n;j++)//if(R[j].key<R[k].key)k=j; //kif(k!=i){// //

在當(dāng)前無序區(qū) R[i‥n]中選 key最小的記錄記下目前找到的最小關(guān)鍵字所在的位置交換 R[i] 和R[k]

R[k]R[0]=R[i];R[i]=R[k];R[k]=R[0];}//endif}//endfor}//========== 大根堆調(diào)整函數(shù) =======voidHeapify(SeqListR,intlow,inthigh){ // 將R[low..high] 調(diào)整為大根堆,除 R[low] 外,其余結(jié)點均滿足堆性質(zhì)intlarge; //large 指向調(diào)整結(jié)點的左、右孩子結(jié)點中關(guān)鍵字較大者RecTypetemp=R[low];// 暫存調(diào)整結(jié)點for(large=2*low;large<=high;large*=2){//R[low] 是當(dāng)前調(diào)整結(jié)點//

large>high,

則表示

R[low]

是葉子,調(diào)整結(jié)束;否則先令

large

指向

R[low]

的左孩子if(large<high&&R[large].key<R[large+1].key)large++; // 若R[low] 的右孩子存在且關(guān)鍵字大于左兄弟,則令

large

指向它//

現(xiàn)在

R[large]

是調(diào)整結(jié)點

R[low]

的左右孩子結(jié)點中關(guān)鍵字較大者if(temp.key>=R[large].key)//temp

始終對應(yīng)

R[low]break;//

當(dāng)前調(diào)整結(jié)點不小于其孩子結(jié)點的關(guān)鍵字,結(jié)束調(diào)整R[low]=R[large];//

相當(dāng)于交換了

R[low]

R[large]low=large;//

low

指向新的調(diào)整結(jié)點,相當(dāng)于

temp

已篩下到

large

的位置}R[low]=temp; // 將被調(diào)整結(jié)點放入最終位置上}//========== 構(gòu)造大根堆 ==========voidBuildHeap(SeqListR){ // 將初始文件 R[1..n] 構(gòu)造為堆inti;for(i=n/2;i>0;i--)Heapify(R,i,n); // 將R[i..n] 調(diào)整為大根堆}//========== 堆排序 ===========voidHeapSort(SeqListR){ // 對R[1..n] 進行堆排序,不妨用

R[0]

做暫存單元inti;BuildHeap(R);//for(i=n;i>1;i--){

//

R[1..n]

構(gòu)造為初始大根堆對當(dāng)前無序區(qū) R[1..i]

進行堆排序,共做

n-1

趟。R[0]=R[1];R[1]=R[i];R[i]=R[0];Heapify(R,1,i-1); //

//

R[1..i-1]

將堆頂和堆中最后一個記錄交換重新調(diào)整為堆,僅有 R[1]可能違反堆性質(zhì)。}}//===== 將兩個有序的子序列 R[low..m]

R[m+1..high]

歸并成有序的序列

R[low..high]==voidMerge(SeqListR,intlow,intm,inthigh){inti=low,j=m+1,p=0;// 置初始值RecType*R1; //R1 為局部量R1=(RecType*)malloc((high-low+1)*sizeof(RecType));// 申請空間while(i<=m&&j<=high) // 兩個子序列非空時取其小者輸出到R1[p++]=(R[i].key<=R[j].key)?R[i++]:R[j++];

R1[p]

上while(i<=m)

//

若第一個子序列非空,則復(fù)制剩余記錄到

R1R1[p++]=R[i++];while(j<=high)

//

若第二個子序列非空,則復(fù)制剩余記錄到

R1中R1[p++]=R[j++];for(p=0,i=low;i<=high;p++,i++)R[i]=R1[p]; // 歸并完成后將結(jié)果復(fù)制回 R[low..high]}//========= 對R[1..n] 做一趟歸并排序voidMergePass(SeqListR,intlength){

========inti;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) // 尚有一個子序列,其中后一個長度小于 lengthMerge(R,i,i+length-1,n);// 歸并最后兩個子序列// 注意:若 i≤n且i+length-1 ≥n時,則剩余一個子序列輪空,無須歸并}//========== 自底向上對 R[1..n] 做二路歸并排序 ===============voidMergeSort(SeqListR){intlength;for(length=1;length<n;length*=2) // 做[lgn] 趟排序MergePass(R,length); // 有序長度≥ n時終止}//========== 輸入順序表 ========voidinput_int(SeqListR){inti;printf("Pleaseinputnum(int):");scanf("%d",&n);printf("Plaseinput%dinteger:",n);for(i=1;i<=n;i++)scanf("%d",&R[i].key);}//========== 輸出順序表 ========voidoutput_int(SeqListR){inti;for(i=1;i<=n;i++)printf("%4d",R[i].key);}//========== 主函數(shù) ======voidmain(){inti;SeqListR;input_int(R);printf("\t********Select**********\n");printf("\t1:InsertSort\n");printf("\t2:BubbleSort\n");printf("\t3:QuickSort\n");printf("\t4:StraightSelectionSort\n");printf("\t5:HeapSort\n");printf("\t6:MergeSort\n");printf("\t7:Exit\n");printf("\t***************************\n");scanf("%d",&i); // 輸入整數(shù)switch(i){case1:InsertSort(R);break; //case2:BubbleSort(R);break;case3:QuickSort(R,1,n);break; //case4:SelectSort(R);break;case5:HeapSort(R);break; //case6:MergeSort(R);break;case7:exit(0); //

1-7//////

,選擇排序方式值為 1,直接插入排序值為2,冒泡法排序值為 3,快速排序值為4,直接選擇排序值為 5,堆排序值為6,歸并排序值為 7,結(jié)束程序}printf("Sortreult:");output_int(R);}實驗結(jié)果:Pleaseinputnum(int):10Plaseinput10integer:26530175112993786374269476438********Select**********InsertSortBubbleSortQ

溫馨提示

  • 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
  • 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
  • 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會有圖紙預(yù)覽,若沒有圖紙預(yù)覽就沒有圖紙。
  • 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
  • 5. 人人文庫網(wǎng)僅提供信息存儲空間,僅對用戶上傳內(nèi)容的表現(xiàn)方式做保護處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負責(zé)。
  • 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請與我們聯(lián)系,我們立即糾正。
  • 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時也不承擔(dān)用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。

評論

0/150

提交評論