版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請進(jìn)行舉報或認(rèn)領(lǐng)
文檔簡介
1、<p><b> 電子與信息工程學(xué)院</b></p><p> 數(shù)據(jù)結(jié)構(gòu)課程設(shè)計報告</p><p> ( 2012——2013年度第一學(xué)期)</p><p> 課程名稱: 數(shù)據(jù)結(jié)構(gòu)課程設(shè)計 </p><p> 題 目 一:6.3“哈夫曼樹”的建立及其應(yīng)用 </p>
2、;<p> 題 目 二: 3.4.6括號匹配的檢驗 </p><p> 院 系: 計算機(jī)科學(xué)系 </p><p> 班 級: </p><p> 姓 名: </p&g
3、t;<p> 學(xué) 號: </p><p> 指導(dǎo)教師: </p><p> 成 績: </p><p> 2012 年 月 日</p><p> 設(shè)計題目<
4、;一>: 6.3“哈夫曼樹”的建立及其應(yīng)用</p><p><b> 一、設(shè)計要求</b></p><p><b> 1.問題描述</b></p><p> 設(shè)有一段電文由字符集{A,B,C,D,E,F,G,H}組成,各字符在電文中出現(xiàn)的次數(shù)集為{5,29,7,8,14,23,3,11},試設(shè)計各字符的哈
5、夫曼編碼。</p><p><b> 2.需求分析</b></p><p> ?。?)設(shè)計哈夫曼樹。具體的構(gòu)造方法如下:以字符集{A,B,C,D,E,F,G,H}作為葉子結(jié)點,以各字符出現(xiàn)的次數(shù){5,29,7,8,14,23,3,11}作為各葉子結(jié)點的權(quán)值構(gòu)造一棵哈夫曼樹。</p><p> ?。?)設(shè)計哈夫曼編碼。按照構(gòu)造出來的哈夫曼樹,規(guī)
6、定哈夫曼樹的左分支為0,右分支為1,則從根結(jié)點到每個葉子結(jié)點所經(jīng)過的分支對應(yīng)的0和1組成的序列便為該結(jié)點對應(yīng)字符的哈夫曼編碼。</p><p><b> 二、概要設(shè)計</b></p><p><b> 1.主界面設(shè)計</b></p><p> 運行界面如圖1所示:</p><p> 圖1哈夫
7、曼編碼主菜單</p><p><b> 2.存儲結(jié)構(gòu)設(shè)計</b></p><p> 對于哈夫曼編碼問題,希望在構(gòu)造哈夫曼樹的同時能方便地實現(xiàn)從雙親結(jié)點到左右孩子結(jié)點的操作,在進(jìn)行哈夫曼編碼時又要求能方便地實現(xiàn)從孩子結(jié)點到雙親結(jié)點的操作。因此,本程序選擇樹的雙親孩子表示法作為哈夫曼樹的存儲結(jié)構(gòu),并加入了指示結(jié)點權(quán)值的信息。</p><p>&
8、lt;b> 3.系統(tǒng)功能設(shè)計</b></p><p> 本程序完成了從哈夫曼樹的構(gòu)造到實現(xiàn)并輸出哈夫曼編碼的過程,分別由兩個子程序完成,其設(shè)計如下:</p><p> ?。?)選擇權(quán)值最小的樹。選擇權(quán)值最小的樹由函數(shù)Select()實現(xiàn)。該功能按照哈夫曼樹的構(gòu)造步驟,在當(dāng)前已構(gòu)成的n(n>=2)棵二叉樹的集合中選取兩棵根結(jié)點權(quán)值最小的樹作為左右子樹構(gòu)造一棵新的二
9、叉樹。</p><p> ?。?)哈夫曼編碼。哈夫曼編碼由函數(shù)HuffmanCoding( )實現(xiàn)。該功能首先調(diào)用函數(shù)Select()實現(xiàn)哈夫曼樹的構(gòu)造,然后從葉子到根逆向根據(jù)哈夫曼編碼的要求,一次求出每個字符的哈夫曼編碼。</p><p><b> 三、模塊設(shè)計</b></p><p><b> 1.模塊設(shè)計 </b>
10、;</p><p> 本程序包含3個模塊:主程序模塊、哈夫曼編碼模塊和選擇模塊。其調(diào)用關(guān)系如圖2所示。</p><p> 圖2 模塊調(diào)用示意圖</p><p> 2.系統(tǒng)子程序及功能設(shè)計</p><p> 本程序共設(shè)置3個子程序,各子程序的函數(shù)名及功能說明如下。</p><p> (1)void Select
11、(HuffmanTree &HT,int m,int *s1,int *s2)</p><p> //選擇權(quán)值最小的兩個結(jié)點</p><p> (2)void HuffmanCoding(HuffmanTree &HT,HuffmanCode &HC,int *w,int n)</p><p><b> //構(gòu)造哈夫曼編碼&
12、lt;/b></p><p> (3)void main( ) //主函數(shù)。輸入結(jié)點個數(shù)及權(quán)值,調(diào)用哈夫曼編碼模塊函數(shù)</p><p> 3.函數(shù)主要調(diào)用關(guān)系圖 </p><p> 本程序3個子程序之間的主要調(diào)用關(guān)系如圖3所示。 </p><p> 圖中數(shù)字是各函數(shù)的編號
13、 </p><p> 圖3系統(tǒng)函數(shù)調(diào)用關(guān)系圖</p><p><b> 四、詳細(xì)設(shè)計</b></p><p><b> 1.?dāng)?shù)據(jù)類型定義</b></p><p> typedef struct</p><p><b> {</b><
14、/p><p> unsigned int weight; //用來存放各個結(jié)點的權(quán)值</p><p> unsigned int parent, lchild, rchild; //指向雙親、孩子結(jié)點的指針</p><p> }HTNode, *HuffmanTree; //
15、動態(tài)分配數(shù)組存儲哈夫曼樹</p><p> typedef char * * HuffmanCode; //動態(tài)分配數(shù)組存儲哈夫曼編碼表</p><p> 2.系統(tǒng)主要子程序詳細(xì)設(shè)計</p><p> 哈夫曼編碼模塊設(shè)計分兩步:首先構(gòu)造哈夫曼樹,然后完成哈夫曼編碼。</p><p> void Huffma
16、nCoding(HuffmanTree &HT,HuffmanCode &HC,int *w,int n)</p><p> {//w存放n個字符的權(quán)值(均>0),構(gòu)造哈夫曼樹HT并求出n個字符的哈夫曼編碼HC</p><p> int i,j,m,s1,s2,start;</p><p><b> char *cd;</
17、b></p><p> unsigned int c,f;</p><p> if (n<=1) return;</p><p> m = 2 * n - 1;</p><p> HT = (HuffmanTree)malloc((m+1) * sizeof(HTNode)); //0號單元未用</p>
18、<p> for (i=1;i<=n;i++) //葉子結(jié)點初始化并放入1-n號單元</p><p><b> {</b></p><p> HT[i].weight=w[i];</p><p> HT[i].parent=0;</p><p> HT[i].lchi
19、ld=0;</p><p> HT[i].rchild=0;</p><p><b> }</b></p><p> for (i=n+1; i<=m;i++) //非葉子結(jié)點初始化</p><p><b> {</b></p><p> HT
20、[i].weight=0;</p><p> HT[i].parent=0;</p><p> HT[i].lchild=0;</p><p> HT[i].rchild=0;</p><p><b> }</b></p><p> printf("\n哈夫曼樹的構(gòu)造過程如下所
21、示:\n");</p><p> printf("HT初態(tài):\n 結(jié)點 weight parent lchild rchild");</p><p> for (i=1;i<=m;i++) //完成構(gòu)造哈夫曼樹算法的第1個步驟</p><p> printf("\n%4d%8d%8d%8d
22、%8d",i,HT[i].weight,HT[i].parent,HT[i].lchild,HT[i].rchild);</p><p> printf(" 按任意鍵,繼續(xù) ...");</p><p><b> getch();</b></p><p> //創(chuàng)建哈夫曼樹HT</p><
23、;p> for (i=n+1;i<=m;i++)</p><p><b> {</b></p><p> Select (HT,i-1,&s1,&s2); </p><p> //在HT[1..i-1]中選擇parent為0且weight最小的兩個結(jié)點</p><p> HT[s
24、1].parent=i;HT[s2].parent=i;</p><p> HT[i].lchild=s1;HT[i].rchild=s2;</p><p> //將選取根結(jié)點權(quán)值最小的樹作為左右子樹</p><p> HT[i].weight=HT[s1].weight+HT[s2].weight;</p><p> //置新二叉樹
25、的根結(jié)點權(quán)值為其左、右子樹根結(jié)點之和</p><p> printf("\nselect:s1=%d s2=%d\n",s1,s2);</p><p> //根結(jié)點權(quán)值最小的樹在HT中的位置</p><p> printf(" 結(jié)點 weight parent lchild rchild");</p>
26、<p> for (j=1;j<=i;j++)</p><p> //輸出選取根結(jié)點權(quán)值最小樹的過程</p><p> printf("\n%4d%8d%8d%8d%8d",j,HT[j].weight,HT[j].parent,HT
27、 [j].lchild,HT[j].rchild);</p><p> printf(" 按任意鍵,繼續(xù) ...");</p><p><b> getch();</b></p><p><b> }</b></p><p> printf(&q
28、uot;\n%d個字符的哈夫曼編碼如下:\n",n);</p><p> //從葉子到根逆向求每個字符的哈夫曼編碼</p><p> HC=(HuffmanCode)malloc((n+1)*sizeof(char *));//分配n個編碼的頭指針</p><p> cd = (char * )malloc(n*sizeof(char));
29、 //分配求編碼的工作空間</p><p> cd[n-1] = '\0'; //編碼結(jié)束符</p><p> for (i=1;i<=n;i++) //逐個字符求哈夫曼編碼</p><p><b> {<
30、/b></p><p> start =n-1; //編碼結(jié)束符位置</p><p> for (c=i,f=HT[i].parent;f!=0;c=f,f=HT[f].parent) </p><p> //葉子結(jié)點到根逆向求編碼</p><p>
31、if (HT[f].lchild==c) cd[--start] ='0';</p><p> else cd[--start] = '1';</p><p> HC[i] = (char *)malloc((n-start)*sizeof(char));</p><p> //為第i個字符編碼分配空間</p>
32、<p> strcpy(HC[i], &cd[start]); //從cd復(fù)制編碼(串)到HC</p><p><b> }</b></p><p> free(cd); //釋放工作空間</p><p> for(i=1;i<=n;i++
33、)</p><p> printf("<%2d>編碼:%s\n",HT[i].weight,HC[i]);</p><p> }//HuffmanCoding</p><p><b> 五、測試分析</b></p><p> 根據(jù)設(shè)計要求中的問題描述分別輸入字符的個數(shù)和對應(yīng)的權(quán)值,
34、程序運行得到圖4的開始界面。</p><p> 圖4哈夫曼編碼程序開始界面</p><p> 構(gòu)造哈夫曼樹的過程如圖(5~ 12)所示。</p><p><b> 圖5</b></p><p><b> 圖6</b></p><p><b> 圖7<
35、/b></p><p><b> 圖8</b></p><p><b> 圖9</b></p><p><b> 圖10</b></p><p><b> 圖11</b></p><p><b> 圖12&
36、lt;/b></p><p> 構(gòu)造哈夫曼編碼如圖13所示。</p><p><b> 圖13 哈夫曼編碼</b></p><p><b> 六、用戶手冊 </b></p><p> ?。?)本程序執(zhí)行文件為“哈夫曼樹的建立及其應(yīng)用.exe”。</p><p>
37、?。?)進(jìn)入本程序之后,分別輸入哈夫曼編碼字符的個數(shù)和對應(yīng)的權(quán)值。</p><p> ?。?)隨即顯示哈夫曼樹的構(gòu)造過程,最終顯示出對應(yīng)權(quán)值的哈夫曼編碼。</p><p><b> 七、調(diào)試報告</b></p><p> 此次課程設(shè)計主要是了解哈夫曼樹的設(shè)計,并學(xué)會哈夫曼編碼的設(shè)計。通過這次課程設(shè)計,我學(xué)到了課本上以外的許多知識,學(xué)會了樹相
38、關(guān)的基礎(chǔ)知識,受益匪淺。</p><p> 《數(shù)據(jù)結(jié)構(gòu)》是一門實踐性較強的課程,為了學(xué)好這門課程,必須在掌握理論知識的同時,加強上機(jī)實踐。其次是,根據(jù)實際問題的需要,對各個方面的優(yōu)缺點加以綜合平衡,從中選擇比較適宜的實現(xiàn)方法。比如在選擇數(shù)據(jù)結(jié)構(gòu)的時候,就要求從時間性能和空間性能去考慮,從而使得能編寫出更加實用和高效的代碼,而要做到這點,就要求對知識點很熟悉。</p><p> 在這次課
39、程設(shè)計中曾遇到了不少問題,比如輸入整型變量的時候,沒有辦法限制其不能輸入字符型,還有類型必須前后匹配等等。使我明白了理論與實際相結(jié)合的重要性,使我更深刻地意識到:掌握了好的理論并不一定能馬上將其運用到自己的程序中,而只有通過不斷地學(xué)習(xí)和實踐,不斷地運用它,才能熟能生巧!</p><p><b> 八、程序清單</b></p><p> #include <s
40、tdio.h></p><p> #include <malloc.h></p><p> #include <string.h></p><p> #include <conio.h></p><p> typedef struct</p><p><b>
41、 {</b></p><p> unsigned int weight; //用來存放各個結(jié)點的權(quán)值</p><p> unsigned int parent,lchild,rchild; //指向雙親、孩子結(jié)點的指針</p><p> }HTNode, *HuffmanTree;
42、 //動態(tài)分配數(shù)組存儲哈夫曼樹</p><p> typedef char * * HuffmanCode; //動態(tài)分配數(shù)組存儲哈夫曼編碼表</p><p> //1.選擇權(quán)值最小的兩個結(jié)點</p><p> void Select(HuffmanTree &HT,int m,int *s1,int *s2)</p>
43、<p><b> {</b></p><p> int i,min;</p><p> for(i=1;i<=m;i++) //在HT[1..i-1]中選擇parent為0且weight最小的兩個結(jié)點</p><p><b> {</b></p><p> if(
44、HT[i].parent==0)</p><p><b> {</b></p><p> min = i;i=m+1;</p><p><b> }</b></p><p><b> }</b></p><p> for(i=1;i<=m
45、;i++) //parent為0且weight最小的兩個結(jié)點,第一個序號為s1</p><p><b> {</b></p><p> if(HT[i].parent == 0)</p><p><b> {</b></p><p> if(HT[i].weight < HT[mi
46、n].weight)</p><p><b> min = i;</b></p><p><b> }</b></p><p><b> }</b></p><p> *s1 = min;</p><p> for(i=1; i<=m;i
47、++) //在HT[1..i-1]中選擇parent為0且weight最小的兩個結(jié)點</p><p><b> {</b></p><p> if(HT[i].parent == 0 &&i!=(*s1))</p><p><b> {</b></p><p> min
48、 = i;i = m+1;</p><p><b> }</b></p><p><b> }</b></p><p> for(i=1;i<=m;i++) //parent為0且weight最小的兩個結(jié)點,第二個序號為s2</p><p><b> {</b&g
49、t;</p><p> if(HT[i].parent == 0 &&i!=(*s1))</p><p><b> {</b></p><p> if(HT[i].weight < HT[min].weight)</p><p><b> min = i;</b><
50、;/p><p><b> }</b></p><p><b> }</b></p><p> *s2 = min;</p><p><b> }//Select</b></p><p> //2.構(gòu)造哈夫曼編碼</p><p&g
51、t; void HuffmanCoding(HuffmanTree &HT,HuffmanCode &HC,int *w,int n)</p><p> {//w存放n個字符的權(quán)值(均>0),構(gòu)造哈夫曼樹HT并求出n個字符的哈夫曼編碼HC</p><p> int i,j,m,s1,s2,start;</p><p><b>
52、 char *cd;</b></p><p> unsigned int c,f;</p><p> if (n<=1) return;</p><p> m = 2 * n - 1;</p><p> HT = (HuffmanTree)malloc((m+1) * sizeof(HTNode)); //0
53、號單元未用</p><p> for (i=1;i<=n;i++) //葉子結(jié)點初始化并放入1-n號單元</p><p><b> {</b></p><p> HT[i].weight=w[i];</p><p> HT[i].parent=0;</p><p&
54、gt; HT[i].lchild=0;</p><p> HT[i].rchild=0;</p><p><b> }</b></p><p> for (i=n+1; i<=m;i++) //非葉子結(jié)點初始化</p><p><b> {</b></p>
55、<p> HT[i].weight=0;</p><p> HT[i].parent=0;</p><p> HT[i].lchild=0;</p><p> HT[i].rchild=0;</p><p><b> }</b></p><p> printf("
56、;\n哈夫曼樹的構(gòu)造過程如下所示:\n");</p><p> printf("HT初態(tài):\n 結(jié)點 weight parent lchild rchild");</p><p> for (i=1;i<=m;i++) //完成構(gòu)造哈夫曼樹算法的第1個步驟</p><p> printf("
57、;\n%4d%8d%8d%8d%8d",i,HT[i].weight,HT[i].parent,HT[i].lchild,HT[i].rchild);</p><p> printf(" 按任意鍵,繼續(xù) ...");</p><p><b> getch();</b></p><p> //創(chuàng)建哈夫曼樹HT
58、</p><p> for (i=n+1;i<=m;i++)</p><p><b> {</b></p><p> Select (HT,i-1,&s1,&s2); </p><p> //在HT[1..i-1]中選擇parent為0且weight最小的兩個結(jié)點</p>
59、<p> HT[s1].parent=i;HT[s2].parent=i;</p><p> HT[i].lchild=s1;HT[i].rchild=s2;</p><p> //將選取根結(jié)點權(quán)值最小的樹作為左右子樹</p><p> HT[i].weight=HT[s1].weight+HT[s2].weight;</p><
60、;p> //置新二叉樹的根結(jié)點權(quán)值為其左、右子樹根結(jié)點之和</p><p> printf("\nselect:s1=%d s2=%d\n",s1,s2);</p><p> //根結(jié)點權(quán)值最小的樹在HT中的位置</p><p> printf(" 結(jié)點 weight parent lchild rchild&qu
61、ot;);</p><p> for (j=1;j<=i;j++)</p><p> //輸出選取根結(jié)點權(quán)值最小樹的過程</p><p> printf("\n%4d%8d%8d%8d%8d",j,HT[j].weight,HT[j].parent,HT
62、 [j].lchild,HT[j].rchild);</p><p> printf(" 按任意鍵,繼續(xù) ...");</p><p><b> getch();</b></p><p><b> }</b></p><p
63、> printf("\n%d個字符的哈夫曼編碼如下:\n",n);</p><p> //從葉子到根逆向求每個字符的哈夫曼編碼</p><p> HC=(HuffmanCode)malloc((n+1)*sizeof(char *));//分配n個編碼的頭指針</p><p> cd = (char * )malloc(n*size
64、of(char)); //分配求編碼的工作空間</p><p> cd[n-1] = '\0'; //編碼結(jié)束符</p><p> for (i=1;i<=n;i++) //逐個字符求哈夫曼編碼</p><p><b&g
65、t; {</b></p><p> start =n-1; //編碼結(jié)束符位置</p><p> for (c=i,f=HT[i].parent;f!=0;c=f,f=HT[f].parent) </p><p> //葉子結(jié)點到根逆向求編碼</p><p
66、> if (HT[f].lchild==c) cd[--start] ='0';</p><p> else cd[--start] = '1';</p><p> HC[i] = (char *)malloc((n-start)*sizeof(char));</p><p> //為第i個字符編碼分配空間</p
67、><p> strcpy(HC[i], &cd[start]); //從cd復(fù)制編碼(串)到HC</p><p><b> }</b></p><p> free(cd); //釋放工作空間</p><p> for(i=1;i<=n;
68、i++)</p><p> printf("<%2d>編碼:%s\n",HT[i].weight,HC[i]);</p><p> }//HuffmanCoding</p><p><b> //3.主函數(shù)</b></p><p> void main()</p>&
69、lt;p><b> {</b></p><p> HuffmanTree myHuffmanTree;</p><p> HuffmanCode myHuffmanCode;</p><p> int *w,i; </p><p>
70、; int n, wei; //編碼個數(shù)及權(quán)值</p><p> printf("請輸入需要哈夫曼編碼的字符個數(shù):");</p><p> scanf("%d",&n);</p><p> w=(int *)malloc((n+1
71、) *sizeof(int));</p><p> for(i=1;i<=n;i++)</p><p><b> {</b></p><p> printf("請輸入第%d字符的權(quán)值:",i);</p><p> fflush(stdin);</p><p>
72、scanf("%d",&wei);</p><p><b> w[i]=wei;</b></p><p><b> }</b></p><p> HuffmanCoding(myHuffmanTree,myHuffmanCode,w,n);</p><p><
73、b> }</b></p><p> 設(shè)計題目<二>: 3.4.6括號匹配的檢驗 </p><p><b> 一、設(shè)計要求</b></p><p><b> 1.問題描述</b></p><p> 假設(shè)表達(dá)式中允許有兩種括號:圓括號和方括號,其
74、嵌套的順序隨意,即(()[ ])或[([ ] [ ])]等為正確格式,[( ])或((( ]均為不正確的格式。檢驗括號是否匹配的方法可用“期待的緊迫程度”這個概念來描述。例如:考慮下列的括號序列:</p><p> [ ( [ ] [ ] ) ]</p><p> 1 2 3 4 5 6 7 8</p><p> 當(dāng)計算機(jī)接受了第1個括號以后,他期待著與其匹配
75、的第8個括號的出現(xiàn),然而等來的卻是第2個括號,此時第1個括號“[”只能暫時靠邊,而迫切等待與第2個括號相匹配的 第7個括號“)”的出現(xiàn),類似的,因只等來了第3個括號“[”,此時,其期待的緊迫程度較第2個括號更緊迫,則第2個括號只能靠邊,讓位于第3個括號,顯然第3個括號的期待緊迫程度高于第2個括號,而第2個括號的期待緊迫程度高于第1個括號;在接受了第4個括號之后,第3個括號的期待得到了滿足,消解之后,第2個括號的期待匹配就成了最急迫的任務(wù)
76、了,…… ,依次類推。可見這個處理過程正好和棧的特點相吻合。</p><p><b> 2.需求分析</b></p><p> 輸入的形式和輸入值的范圍:從鍵盤上以字符串的形式輸入括號序列。</p><p> 輸出的形式:括號匹配或是括號不匹配。</p><p> 程序所能達(dá)到的功能:檢驗括號是否匹配。</
77、p><p> 測試數(shù)據(jù):輸入([ ]()),結(jié)果“匹配”, 輸入 [(( )],結(jié)果“此串括號匹配不合法”</p><p><b> 二、概要設(shè)計</b></p><p><b> 1.主界面設(shè)計</b></p><p> 運行界面如圖1所示:</p><
78、;p> 圖1 括號匹配的檢驗主菜單</p><p><b> 2.存儲結(jié)構(gòu)設(shè)計</b></p><p> 對于括號匹配的檢驗問題,希望在輸入一個括號序列后,程序能檢測出該括號序列是否匹配,則設(shè)置一個棧來實現(xiàn)。當(dāng)遇到 ( 或 [ 時進(jìn)棧,遇到 ) 或 ] 時出棧進(jìn)行匹配檢驗,如果出現(xiàn)不匹配的情況立即結(jié)束,否則繼續(xù)取下一個字符。如果沒有遇到不匹配的情況,最后判
79、斷棧是否為空,棧為空,括號匹配,否則不匹配。</p><p><b> 3.系統(tǒng)功能設(shè)計</b></p><p> 本程序完成了輸入括號序列后檢驗括號序列是否匹配,定義棧結(jié)構(gòu)和判斷棧的情況有以下說明:</p><p> typedef struct{ }定義棧結(jié)構(gòu)體</p><p> Status CreatS
80、tack(SqStack &S)</p><p> 初始條件:棧指針已存在</p><p> 操作結(jié)果:定義空棧并分配存儲空間,成功返回ok</p><p> Status StackEmpty(SqStack S)</p><p><b> 初始條件:棧已存在</b></p><p&
81、gt; 操作結(jié)果:判斷是否為空,是返回ok</p><p> Status Push(SqStack &S,Elem e)</p><p> 初始條件:棧已存在,e已知</p><p> 操作結(jié)果:將e壓入棧中,成功返回ok</p><p> Status Pop(SqStack &S,Elem &e)<
82、;/p><p> 初始條件:棧非空,棧頂元素等于e</p><p> 操作結(jié)果:棧頂元素出棧</p><p> Status Bracket(SqStack &S,char *str)</p><p> 初始條件:空棧已存在,括號串非空</p><p> 操作結(jié)果:輸出括號串是否匹配</p>
83、<p> void main()</p><p> 操作結(jié)果:在屏幕上顯示操作菜單</p><p><b> 三、模塊設(shè)計</b></p><p><b> 1.模塊設(shè)計 </b></p><p> 本程序包含2個模塊:主程序模塊和棧操作模塊。其調(diào)用關(guān)系如圖2所示。</
84、p><p> 圖2 模塊調(diào)用示意圖</p><p> 2.系統(tǒng)子程序及功能設(shè)計</p><p> 本程序共設(shè)置6個子程序,各子程序的函數(shù)名及功能說明如下。</p><p> (1)Status CreatStack(SqStack &S)//創(chuàng)建空堆棧,棧頂指針和棧底指針相等時,棧為空</p><p>
85、 (2)Status StackEmpty(SqStack S)//堆棧是否為空</p><p> (3)Status Push(SqStack &S,Elem e)//進(jìn)棧</p><p> (4)Status Pop(SqStack &S,Elem &e)//出棧</p><p> (5)Status Bracket(SqStack
86、 &S,char *str)//檢驗括號匹配</p><p> (6)void main( ) //主函數(shù)。輸入括號序列,調(diào)用棧操作模塊函數(shù)</p><p> 3.函數(shù)主要調(diào)用關(guān)系圖</p><p> 本程序6個子程序之間的主要調(diào)用關(guān)系如圖3所示。</p><p> 圖3 系統(tǒng)函數(shù)調(diào)用關(guān)系圖</p&
87、gt;<p><b> 四、詳細(xì)設(shè)計</b></p><p><b> 1.?dāng)?shù)據(jù)類型定義</b></p><p> typedef struct{ </p><p> Elem *base; //棧底指針</p><p> Elem *top; //棧頂指針 </p&g
88、t;<p> int size; //當(dāng)前已分配的存儲空間</p><p><b> }SqStack;</b></p><p> typedef int Status;</p><p> 2.系統(tǒng)主要子程序詳細(xì)設(shè)計</p><p> Status CreatStack(SqStack &
89、S)//創(chuàng)建空堆棧,棧頂指針和棧底指針相等時,棧為空</p><p><b> { </b></p><p> S.base=(Elem *)malloc(STACK_SIZE*sizeof(Elem)); </p><p> S.top=S.base; S.size=STACK_SIZE; return OK;</p>&
90、lt;p> } //棧頂指針和棧底指針相等,創(chuàng)建空堆棧</p><p> Status StackEmpty(SqStack S) //堆棧是否為空</p><p><b> { </b></p><p> if(S.top!=S.base) </p><p> return ER
91、ROR; //棧頂指針和棧底指針不相等,則返回ERROR</p><p> return OK; </p><p><b> }</b></p><p> Status Push(SqStack &S,Elem e)//進(jìn)棧</p><p><b> { </b
92、></p><p> if(S.top-S.base>=S.size)</p><p> { //棧滿,追加存儲空間 </p><p> S.base=(Elem *)realloc(S.base,(S.size+STACK_INC)*sizeof(Elem));</p><p> S.top=S.base+S.size
93、; S.size+=STACK_INC;</p><p><b> } </b></p><p> *S.top=e; //將e賦給棧頂指針</p><p> S.top+=1; //棧頂指針向上移加1</p><p> return OK;}</p><p>
94、; Status Pop(SqStack &S,Elem &e)//出棧</p><p><b> { </b></p><p> if(S.top==S.base) </p><p> return ERROR; //判斷棧是否為空,空則返回ERROR</p><p> S.
95、top-=1; //否則棧頂指針下移減1</p><p> e=*S.top; //將站頂元素賦給e</p><p> return OK;</p><p><b> }</b></p><p> Status Bracket(SqStack &S,char *
96、str)//檢驗括號匹配</p><p><b> { </b></p><p> int i=0,flag1=0,flag2;</p><p><b> Elem e; </b></p><p> while(str[i]!='\0') //括號序列不為空</p
97、><p><b> { </b></p><p> switch(str[i])</p><p><b> { </b></p><p> case '(':Push(S,'(');</p><p> break; //'(
98、'進(jìn)棧 </p><p> case '[':Push(S,'[');</p><p> break; //'['進(jìn)棧</p><p> case ')':{Pop(S,e); </p><p> if(e!='(') </p>
99、<p> flag1=1; //出棧,判斷是否為'(',不是則用flag1標(biāo)記為1</p><p><b> break;</b></p><p><b> } </b></p><p> case ']':{Pop(S,e);</p><p>
100、; if(e!='[') </p><p> flag1=1; //出棧,判斷是否為['',不是則用flag1標(biāo)記為1</p><p><b> break;</b></p><p><b> } </b></p><p> default: brea
101、k; </p><p><b> } </b></p><p> if(flag1) </p><p> break; //出現(xiàn)不匹配,立即結(jié)束循環(huán) </p><p><b> i++; </b></p><p><b> } </b>&l
102、t;/p><p> flag2=StackEmpty(S); //flag2判斷堆棧是否為空 </p><p> if(!flag1 && flag2) </p><p> printf("匹配\n");</p><p><b> else </b></p><
103、p> printf("此串括號匹配不合法\n");</p><p> return OK;</p><p><b> }</b></p><p><b> 五、測試分析</b></p><p> 程序的測試分析如圖4所示。</p><p>
104、<b> 圖4 程序運行圖</b></p><p><b> 六、用戶手冊 </b></p><p> ?。?)本程序執(zhí)行文件為“括號匹配的檢驗.exe”。</p><p> ?。?)進(jìn)入本程序之后,輸入要匹配的括號序列。</p><p> ?。?)顯示“匹配”或“此串括號匹配不合法”。<
105、/p><p><b> 七、調(diào)試報告</b></p><p> 以前做實驗題目的時候總是感覺很難,因為根本就不知道從哪里開始。這次課程設(shè)計讓我對編程有了新的認(rèn)識。</p><p> 拿到題目的時候也是很困惑,后來看了很多有關(guān)的例子,仔細(xì)看了書上關(guān)于棧的算法的知識,覺得就是上課講到的一些內(nèi)容,不是題目難,是自己先把自己嚇住了。</p>
106、;<p> 后來,參照書上的諸多例子,一個模塊一個模塊的編寫,調(diào)試,一個功能一個功能去完善。發(fā)現(xiàn)越做越順利,加上以前的實驗中對于改錯的經(jīng)驗積累和幾個學(xué)得不錯的同學(xué)的幫助,我的程序中的錯誤也一個一個的順利解決。</p><p> 再后來,等我的程序完全做好以后,我竟然可以獨立的幫同學(xué)修改一些以前根本不知所以然的錯誤,其實,從這次實驗中我認(rèn)識到,我要學(xué)習(xí)的還有很多,編程有很多的樂趣也有很多的技巧性和
107、知識性。我將在以后的日子里繼續(xù)認(rèn)真的學(xué)習(xí)知識,積累經(jīng)驗,讓自己的編程能力提高。</p><p><b> 八、程序清單</b></p><p> #include <stdio.h></p><p> #include <malloc.h></p><p> #define OK 1<
108、;/p><p> #define ERROR 0 //定義順序堆棧</p><p> #define STACK_SIZE 100 //存儲空間初始分配量</p><p> #define STACK_INC 10 //存儲空間分配增量</p><p> typedef char Elem;</p><p>
109、typedef struct{ </p><p> Elem *base; //棧底指針</p><p> Elem *top; //棧頂指針 </p><p> int size; //當(dāng)前已分配的存儲空間</p><p><b> }SqStack;</b></p><p> typ
110、edef int Status;</p><p> Status CreatStack(SqStack &S)//創(chuàng)建空堆棧,棧頂指針和棧底指針相等時,棧為空</p><p><b> { </b></p><p> S.base=(Elem *)malloc(STACK_SIZE*sizeof(Elem)); </p>
111、;<p> S.top=S.base; S.size=STACK_SIZE; return OK;</p><p> } //棧頂指針和棧底指針相等,創(chuàng)建空堆棧</p><p> Status StackEmpty(SqStack S) //堆棧是否為空</p><p><b> { </b><
112、;/p><p> if(S.top!=S.base) </p><p> return ERROR; //棧頂指針和棧底指針不相等,則返回ERROR</p><p> return OK; </p><p><b> }</b></p><p> Status P
113、ush(SqStack &S,Elem e)//進(jìn)棧</p><p><b> { </b></p><p> if(S.top-S.base>=S.size)</p><p> { //棧滿,追加存儲空間 </p><p> S.base=(Elem *)realloc(S.base,(S.si
114、ze+STACK_INC)*sizeof(Elem));</p><p> S.top=S.base+S.size; S.size+=STACK_INC;</p><p><b> } </b></p><p> *S.top=e; //將e賦給棧頂指針</p><p> S.top+=1;
115、 //棧頂指針向上移加1</p><p> return OK;}</p><p> Status Pop(SqStack &S,Elem &e)//出棧</p><p><b> { </b></p><p> if(S.top==S.base) </p><
116、p> return ERROR; //判斷棧是否為空,空則返回ERROR</p><p> S.top-=1; //否則棧頂指針下移減1</p><p> e=*S.top; //將站頂元素賦給e</p><p> return OK;</p><p><b>
117、}</b></p><p> Status Bracket(SqStack &S,char *str)//檢驗括號匹配</p><p><b> { </b></p><p> int i=0,flag1=0,flag2;</p><p><b> Elem e; </b>
118、;</p><p> while(str[i]!='\0') //括號序列不為空</p><p><b> { </b></p><p> switch(str[i])</p><p><b> { </b></p><p> case
119、9;(':Push(S,'(');</p><p> break; //'('進(jìn)棧 </p><p> case '[':Push(S,'[');</p><p> break; //'['進(jìn)棧</p><p> case ')&
120、#39;:{Pop(S,e); </p><p> if(e!='(') </p><p> flag1=1; //出棧,判斷是否為'(',不是則用flag1標(biāo)記為1</p><p><b> break;</b></p><p><b> } </b>
121、</p><p> case ']':{Pop(S,e);</p><p> if(e!='[') </p><p> flag1=1; //出棧,判斷是否為['',不是則用flag1標(biāo)記為1</p><p><b> break;</b></p>
122、<p><b> } </b></p><p> default: break; </p><p><b> } </b></p><p> if(flag1) </p><p> break; //出現(xiàn)不匹配,立即結(jié)束循環(huán) </p><p>&l
123、t;b> i++; </b></p><p><b> } </b></p><p> flag2=StackEmpty(S); //flag2判斷堆棧是否為空 </p><p> if(!flag1 && flag2) </p><p> printf("匹配\n&
124、quot;);</p><p><b> else </b></p><p> printf("此串括號匹配不合法\n");</p><p> return OK;</p><p><b> }</b></p><p> void main(){
125、 //主函數(shù)</p><p> char temp,flag='y'; </p><p> while(flag=='y'){</p><p> char str[255];</p><p> SqStack S;</p><p> printf("請輸入要匹配
126、的括號序列:\n");</p><p> scanf("%s",str); </p><p> scanf("%c",&temp); //接受輸入的回車鍵</p><p> CreatStack(S); </p><p> Bracket(S,str);</p>
127、<p> printf("您想再試一次嗎(按y繼續(xù)): ");</p><p> scanf("%c",&flag);</p><p> printf("\n"); </p><p><b> } </b></p><p> prin
溫馨提示
- 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. 眾賞文庫僅提供信息存儲空間,僅對用戶上傳內(nèi)容的表現(xiàn)方式做保護(hù)處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負(fù)責(zé)。
- 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請與我們聯(lián)系,我們立即糾正。
- 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時也不承擔(dān)用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。
最新文檔
- 數(shù)據(jù)結(jié)構(gòu)課程設(shè)計報告 (2)
- 數(shù)據(jù)結(jié)構(gòu)課程設(shè)計報告 (2)
- 數(shù)據(jù)結(jié)構(gòu)課程設(shè)計報告
- 數(shù)據(jù)結(jié)構(gòu)課程設(shè)計報告
- 數(shù)據(jù)結(jié)構(gòu)課程設(shè)計報告
- 數(shù)據(jù)結(jié)構(gòu)課程設(shè)計報告
- 數(shù)據(jù)結(jié)構(gòu)課程設(shè)計報告
- 數(shù)據(jù)結(jié)構(gòu)課程設(shè)計報告
- 數(shù)據(jù)結(jié)構(gòu)課程設(shè)計報告
- 數(shù)據(jù)結(jié)構(gòu)課程設(shè)計報告
- 數(shù)據(jù)結(jié)構(gòu)課程設(shè)計報告
- 《數(shù)據(jù)結(jié)構(gòu)》課程設(shè)計報告
- 數(shù)據(jù)結(jié)構(gòu)課程設(shè)計報告
- 數(shù)據(jù)結(jié)構(gòu)課程設(shè)計 (2)
- 數(shù)據(jù)結(jié)構(gòu)課程設(shè)計--數(shù)據(jù)結(jié)構(gòu)課程設(shè)計----huffman編碼
- 數(shù)據(jù)結(jié)構(gòu)課程設(shè)計——課程設(shè)計報告模板
- 數(shù)據(jù)結(jié)構(gòu)課程設(shè)計報告 (4)
- 數(shù)據(jù)結(jié)構(gòu)課程設(shè)計實習(xí)報告
- 數(shù)據(jù)結(jié)構(gòu)課程設(shè)計報告.doc
- 數(shù)據(jù)結(jié)構(gòu)課程設(shè)計報告 (3)
評論
0/150
提交評論