《計算機軟件技術(shù)基礎 習題一解答.doc》由會員分享,可在線閱讀,更多相關《計算機軟件技術(shù)基礎 習題一解答.doc(17頁珍藏版)》請在裝配圖網(wǎng)上搜索。
線性結(jié)構(gòu) 習題解答
習題解答
3.設n為正整數(shù), 分析下列各程序段中加下劃線的語句的執(zhí)行次數(shù)。
(1) for (int i = 1; i <= n; i++)
for (int j = 1; j <= n; j++) {
c[i][j] = 0.0;
for (int k = 1; k <= n; k++)
c[i][j] = c[i][j] + a[i][k] * b[k][j];
}
(2) x = 0; y = 0;
for (int i = 1; i <= n; i++)
for (int j = 1; j <= i; j++)
for (int k = 1; k <= j; k++)
x = x + y;
(3) int i = 1, j = 1;
while (i<=n && j<=n) {
i = i + 1; j = j + i;
}
(4)* int i =1;
do{
for (int j = 1; j <= n; j++)
i = i + j;
}while(i<100 + n);
【解答】
(1)
(2)
(3)i = 1時,i = 2,j = j + i = 1 + 2 = 2 + 1,
i = 2時,i = 3,j = j + i = ( 2 + 1 ) + 3 = 3 + 1 + 2,
i = 3時,i = 4,j = j + i = ( 3 + 1 + 2 ) + 4 = 4 + 1 + 2 + 3,
i = 4時,i = 5,j = j + i = ( 4 + 1 + 2 + 3 ) + 5 = 5 + 1 + 2 + 3 + 4,
……
i = k時,i = k + 1,j = j + i = ( k + 1 ) + ( 1 + 2 + 3 + 4 + … + k ),
解出滿足上述不等式的k值,即為語句i = i + 1的程序步數(shù)。 (4) for語句每執(zhí)行一次,語句i=i+j將執(zhí)行n次,而i的值會增加
因此,當for語句執(zhí)行k次后,i的值將變?yōu)?
故最終for語句的執(zhí)行次數(shù)k為滿足
的最小整數(shù)k,語句i = i + j的程序步數(shù)為n*k。
4.試編寫一個函數(shù)計算n!*2n的值,結(jié)果存放于數(shù)組A[arraySize]的第n個數(shù)組元素中,0 £ n £ arraySize。若設計算機中允許的整數(shù)的最大值為maxInt,則當n>arraySize或者對于某一個k (0 £ k £ n),使得k!*2k > maxInt時,應按出錯處理??捎腥缦氯N不同的出錯處理方式:
(1) 用printf顯示錯誤信息及exit(1)語句來終止執(zhí)行并報告錯誤;
(2) 用返回整數(shù)函數(shù)值0, 1來實現(xiàn)算法,以區(qū)別是正常返回還是錯誤返回;
(3) 在函數(shù)的參數(shù)表設置一個引用型的整型變量來區(qū)別是正常返回還是某種錯誤返回。
試討論這三種方法各自的優(yōu)缺點,并以你認為是最好的方式實現(xiàn)它。
【解答】
#include
const int arraySize = 100;
const int MaxInt = 0x7fffffff;
int calc( int T[ ], int n ) {
int i, value = 1;T[0]=1;
if ( n != 0 ) {
int edge = MaxInt / n / 2;
for ( i = 1; i < n; i++ ) {
value *= i*2;
T[i] = value;
if ( value > edge ) return 0;
}
value *= n * 2;
}
T[n] = value;
printf("A[ %d ]= %d\n”,n,T[n]);
return 1;
}
void main ( ) {
int A[arraySize];
int i;
for ( i = 0; i < arraySize; i++ )
if ( !calc ( A, i ) ) {
printf("failed at %d .\n", i );
break;
}
}
/*---------順序表結(jié)構(gòu)的定義.為簡化起見,表元素我們使用整型數(shù)據(jù)-----------
-----------數(shù)據(jù)元素從data[0]處開始存儲---------------------------------*/
typedef struct /*注意typedef的使用*/
{
int data[MAXSIZE]; /*數(shù)據(jù)域*/
int length; /*表長*/
}listtype;
5.設有一個線性表 (a0, a1, …, an-2, an-1) 存放在一個一維數(shù)組A[arraySize]中的前n個數(shù)組元素位置。請編寫一個函數(shù)將這個線性表原地逆置,即將數(shù)組的前n個原址內(nèi)容置換為 (an-1, an-2, …, a1, a0)。最后分析此算法的時間復雜度及空間復雜度。
【解答】
void inverse (listtype * A) {
int tmp;
int n= A->length;
for( int i = 0; i <= ( n-1 ) / 2; i++ ){
tmp = A->data[i]; A->data[i] = A->data[n-i-1]; A->data[n-i-1] = tmp;
}
}
時間復雜度:需進行n/2次循環(huán),因此時間復雜度為O(n);
空間復雜度:使用一個整形輔助存儲單元tmp,因此空間復雜度為O(1)。
6.順序表的插入和刪除要求仍然保持各個元素原來的次序。設在等概率情形下, 對有127個元素的順序表進行插入, 平均需要移動多少個元素? 刪除一個元素, 又平均需要移動多少個元素?
【解答】
若設順序表中已有n個元素。又設插入或刪除表中各個元素的概率相等,則在插入時因有n+1個插入位置(可以在表中最后一個表項后面追加),每個元素位置插入的概率為1/(n+1),但在刪除時只能在已有n個表項范圍內(nèi)刪除,所以每個元素位置刪除的概率為1/n。
插入時平均移動元素個數(shù)AMN(Averagy Moving Number )為
刪除時平均移動元素個數(shù)AMN為
7.利用順序表的操作,實現(xiàn)以下的函數(shù)。并分析你所編制的函數(shù)的時間復雜度,并分析(2)與(3)的時間復雜度出現(xiàn)差異的原因。
(1) 從順序表中刪除具有給定值x的所有元素。
(2) 從順序表中刪除其值在給定值s與t之間(要求s小于t)的所有元素。
(3) 從有序順序表中刪除其值在給定值s與t之間(要求s小于t)的所有元素。
(4) 將兩個有序順序表la,lb合并成一個新的有序順序表lc。
(5) 從順序表中刪除所有其值重復的元素,使表中所有元素的值均不相同。
【解答】
(1) 從順序表中刪除具有給定值x的所有元素。
void DelValue(listtype * L, int x ){
int i = 0, j;
while ( i < L->length ) /*循環(huán), 尋找具有值x的元素并刪除它*/
if (L->data[i] == x ) { /*刪除具有值x的元素, 后續(xù)元素前移*/
for (j = i;j < L->length-1; j++ ) L->data[j] = L->data[j+1];
L-length--; /*表長減1*/
}
else i++;
}
(2) 實現(xiàn)刪除其值在給定值s與t之間(要求s小于t)的所有元素的函數(shù)如下:
void DelValue_s_to_t (listtype *L,int s, int t){
int i,j;
if ( L->length == 0 || s >= t ) {
printf(“List is empty or parameters are illegal!\n”); exit(1); }
i = 0;
while ( i < L->length) /*循環(huán), 尋找具有值x的元素并刪除它*/
if (L->data[i]>=s &&L->data[i]<= t){
/*刪除滿足條件的元素, 后續(xù)元素前移*/
for ( j = i; j < L->length-1; j++ ) L->data[j] = L->data[j+1];
L->length--; /*表長減1*/
}
else i++;
}
(3) 實現(xiàn)從有序順序表中刪除其值在給定值s與t之間的所有元素的函數(shù)如下:
void DelValue_s_to_t_1 (listtype *L,int s int t){
int i,j,k;
if ( L->length == 0 || s >= t ){
printf(“List is empty or parameters are illegal!\n”); exit(1); }
for (i = 0; i < L->length; i++ ) /*循環(huán), 尋找值 ≥s 的第一個元素*/
if ( L->data[i] >= s ) break; /*退出循環(huán)時, i指向該元素*/
if ( i < L->length ) {
for (j = 1; i + j < L->length; j++ )/*循環(huán), 尋找值 > t 的第一個元素*/
if (L->data[i+j] > t ) break; /*退出循環(huán)時, i+j指向該元素*/
for (k = i+j; k < L->length; k++ ) /*刪除滿足條件的元素, 后續(xù)元素前移*/
L->data[k-j] = L->data[k];
L->length-= j; /*表長減j*/
}
}
(4) 實現(xiàn)將兩個有序順序表合并成一個新的有序順序表的函數(shù)如下:
listtype * Merge(listtype *LA,listtype *LB ){
/*合并有序順序表LA與LB成為一個新的有序順序表并由函數(shù)返回
listtype *LC;
int value1,value2;
int i,j,k;
initiatelist(LC);
if (LA->length + LB->length > MAXSIZE) {
printf(“表上溢/n”; exit(1);
}
i = 0, j = 0, k = 0;
value1 = LA->data[i];
value2 = LB->data[j];
while (i < LA->length && j < LB->length ) {
/*循環(huán), 兩兩比較, 小者存入結(jié)果表*/
if (value1 <= value2){
LC->data[k] = value1; i++; value1=LA->data[i];}
else {
LC->data[k] = value2; j++; value2=LB->data[j];}
k++;
}
while( i < LA->length){ /*當LA表未檢測完, 繼續(xù)向結(jié)果表傳送*/
LC->data[k] = value1; i++; k++; value1 = LA->data[i];
}
while( j < LB->length){ /*當LB表未檢測完, 繼續(xù)向結(jié)果表傳送*/
LC->data[k] = value2; j++; k++; value2 = LB->data[j];
}
LC->length = k;
return LC;
}
(5) 實現(xiàn)從表中刪除所有其值重復的元素的函數(shù)如下:
void DelDouble(listtype *L){
int i,j,k;
int tmp;
if(L->length == 0 ){
printf(“表空\n”; exit(1);
}
i=0;
while ( i < L->length ) { /*循環(huán)檢測*/
j = i + 1;
tmp = L->data[i];
while( j < L->length ) { /*對于每一個i, 重復檢測一遍后續(xù)元素*/
if( tmp == L->data[j]) { /*如果相等,刪除此結(jié)點,后續(xù)元素前移*/
for( k = j+1; k < L->length; k++ ) L->data[k-1] = L->data[k];
L->length--; /*表最后元素位置減1*/
}
else j++;
}
i++; /*檢測完L->data[i], 檢測下一個*/
}
}
8.線性表可用順序表或鏈表存儲。試問:
(1) 兩種存儲表示各有哪些主要優(yōu)缺點?
(2) 如果有n個表同時并存,并且在處理過程中各表的長度會動態(tài)發(fā)生變化,表的總數(shù)也可能自動改變、在此情況下,應選用哪種存儲表示?為什么?
(3) 若表的總數(shù)基本穩(wěn)定,且很少進行插入和刪除,但要求以最快的速度存取表中的元素,這時,應采用哪種存儲表示?為什么?
【解答】
(1) 順序存儲表示是將數(shù)據(jù)元素存放于一個連續(xù)的存儲空間中,實現(xiàn)順序存取或(按下標)直接存取。它的存儲效率高,存取速度快。但它的空間大小一經(jīng)定義,在程序整個運行期間不會發(fā)生改變,因此,不易擴充。同時,由于在插入或刪除時,為保持原有次序,平均需要移動一半(或近一半)元素,修改效率不高。
鏈接存儲表示的存儲空間一般在程序的運行過程中動態(tài)分配和釋放,且只要存儲器中還有空間,就不會產(chǎn)生存儲溢出的問題。同時在插入和刪除時不需要保持數(shù)據(jù)元素原來的物理順序,只需要保持原來的邏輯順序,因此不必移動數(shù)據(jù),只需修改它們的鏈接指針,修改效率較高。但存取表中的數(shù)據(jù)元素時,只能循鏈順序訪問,因此存取效率不高。
(2) 如果有n個表同時并存,并且在處理過程中各表的長度會動態(tài)發(fā)生變化,表的總數(shù)也可能自動改變、在此情況下,應選用鏈接存儲表示。
如果采用順序存儲表示,必須在一個連續(xù)的可用空間中為這n個表分配空間。初始時因不知道哪個表增長得快,必須平均分配空間。在程序運行過程中,有的表占用的空間增長得快,有的表占用的空間增長得慢;有的表很快就用完了分配給它的空間,有的表才用了少量的空間,在進行元素的插入時就必須成片地移動其他的表的空間,以空出位置進行插入;在元素刪除時,為填補空白,也可能移動許多元素。這個處理過程極其繁瑣和低效。
如果采用鏈接存儲表示,一個表的存儲空間可以連續(xù),可以不連續(xù)。表的增長通過動態(tài)存儲分配解決,只要存儲器未滿,就不會有表溢出的問題;表的收縮可以通過動態(tài)存儲釋放實現(xiàn),釋放的空間還可以在以后動態(tài)分配給其他的存儲申請要求,非常靈活方便。對于n個表(包括表的總數(shù)可能變化)共存的情形,處理十分簡便和快捷。所以選用鏈接存儲表示較好。
(3) 應采用順序存儲表示。因為順序存儲表示的存取速度快,但修改效率低。若表的總數(shù)基本穩(wěn)定,且很少進行插入和刪除,但要求以最快的速度存取表中的元素,這時采用順序存儲表示較好。
/*---------鏈表結(jié)構(gòu)的定義.為簡化起見,表元素我們使用整型數(shù)據(jù)-----------
-----------此鏈表帶頭結(jié)點--------------------------------------------*/
typedef struct mynode{
int data; /*數(shù)據(jù)域:整型*/
struct mynode *next; /*指針域*/
} node, linklisttype;
9.試寫出計算線性鏈表長度的算法。
int lengthsl(linklisttype *L){
linklisttype * p;
int len;
if(L == NULL){return –1;}
p = L->next; /* p指向鏈表L的頭結(jié)點*/
len = 0;
while(p != NULL){
len++;
p = p->next;
}
return len;
}
10.設有一個表頭指針為h的單鏈表。試設計一個算法,通過遍歷一趟鏈表,將鏈表中所有結(jié)點的鏈接方向逆轉(zhuǎn),如下圖所示。要求逆轉(zhuǎn)結(jié)果鏈表的表頭指針h指向原鏈表的最后一個結(jié)點。
【解答】
void LinkListInverse(linklisttype *L){
linklisttype *p, *pre, *next;
if(L == NULL || L->next == NULL ) return; /*表未初始化,或為空表*/
p = L->next;
pre = L;
while( p != NULL ) { /*循環(huán)修改鏈表中所有結(jié)點的后繼指針的指向*/
next = p->next; /*取當前結(jié)點的后繼指針*/
p->next = pre; /*當前結(jié)點的后繼改為指向其原來的前驅(qū)*/
pre = p , p=next; /*指針后移*/
}
L->next->next = NULL; /*原第一個結(jié)點改為最后一個結(jié)點*/
L->next = pre; /*鏈表的頭結(jié)點指向原最后一個結(jié)點*/
}
11.設有一線性鏈表,其結(jié)點值均為整數(shù)。試將該線性鏈表分解為兩個線性鏈表,其中一鏈表中的結(jié)點值均為負整數(shù),而另一鏈表中結(jié)點的值均為正整數(shù),并刪除結(jié)點值為零的結(jié)點。
【解答】
void LinkListDispose(linklisttype * L,linklisttype * LA,linklisttype * LB){
/*
將鏈表L分解為LA、LB兩個鏈表,其中LA中結(jié)點值均為正,LB中結(jié)點值均為負,
并刪除結(jié)點值為零的結(jié)點,最后釋放L的頭結(jié)點。
*/
linklisttype *p , *pt , *pa, * pb;
LA = initiatesl();
pa = LA; /*指向LA中的最后一個結(jié)點*/
LB = initiatesl();
pb = LB; /*指向LB中的最后一個結(jié)點*/
If(L == NULL || L->next == NUUL) return; /*L為空指針或L為空表*/
p = L->next; /*p指向鏈表L的第一個結(jié)點*/
while(p != NULL) /*遍歷鏈表L*/
if(p->data > 0){ /*將p鏈入鏈表LA的pa后*/
pa->next = p;
pa = p;
p = p->next;
}
elseif(p->data < 0){ /*將p鏈入鏈表LB的pb后*/
pb->next = p;
pb = p;
p = p->next;
}
else{ /*刪除值為0的結(jié)點*/
pt = p->next; /*記錄當前結(jié)點的后繼,以便刪除當前結(jié)點*/
free(p);
p = pt;
}
pa->next = NULL;
pb->next = NULL;
free(L);
}
12.設ha和hb分別是兩個帶表頭結(jié)點的非遞減有序單鏈表的表頭指針, 試設計一個算法, 將這兩個有序鏈表合并成一個非遞減有序的單鏈表。要求結(jié)果鏈表仍使用原來兩個鏈表的存儲空間, 不另外占用其它的存儲空間。表中允許有重復的數(shù)據(jù)。
【解答】
void linklistMerge(linklisttype *LA,linklisttype *LB ){
/*合并有序鏈表LA與LB,結(jié)果存入LA中,并釋放LB的頭結(jié)點 */
linklisttype *pa, *pb , *pre ,*pn;
if(LA == NULL || LB == NULL ||) return;
if(LA->next == NULL){ /*LA為空表,則直接將LB鏈入LA即可*/
LA->next = LB->next;
free(LB);
retrun;
}
if(LB->next == NULL) return; /*LB為空表,則直接返回即可*/
pa = LA->next, pb = LB->next ,pre=LA;
while (pa != NULL && pb != NULL) /*循環(huán), 兩兩比較, 小者存入結(jié)果表*/
if(pa->data > pb->next){ /*將pb鏈入LA,然后pb指針后移*/
pre->next = pb;
pn = pb->next;
pb->next = pa;
pb = pn;
pre = pre->next
}
else{ /*pa指針后移*/
pa = pa->next;
pre = pre->next;
}
if(pb != NULL) /*將pb剩余結(jié)點鏈入LA*/
pre->next = pb;
free(LB);
}
13.設ha和hb分別是兩個帶表頭結(jié)點的非遞減有序單鏈表的表頭指針, 試設計一個算法, 將這兩個有序鏈表合并成一個非遞增有序的單鏈表。要求結(jié)果鏈表仍使用原來兩個鏈表的存儲空間, 不另外占用其它的存儲空間。表中允許有重復的數(shù)據(jù)。
【解答】
Linklisttype * linklistMerge_inverse(linklisttype *LA,linklisttype *LB ){
/*合并有序鏈表LA與LB,結(jié)果存入LC中,并釋放LA、LB的頭結(jié)點 ,函數(shù)返回LC*/
linklisttype *pa, *pb , *p;
if(LA == NULL || LB == NULL ||) return;
LC = initiatesl();
pa = LA->next, pb = LB->next;
while (pa != NULL && pb != NULL) /*循環(huán), 兩兩比較, 大者存入LC*/
if(pa->data > pb->next){ /*將pa鏈為LC的第一個結(jié)點*/
p = pa->next;
pa->next = LC->next;
LC->next = pa;
pa = p;
}
else{ /*將pa鏈為LC的第一個結(jié)點*/
p = pb->next;
pb->next = LC->next;
LC->next = pb;
pb = p;
}
while(pa != NULL){ /*將pa剩余結(jié)點鏈入LC*/
p = pa->next;
pa->next = LC->next;
LC->next = pa;
pa = p;
}
while(pb != NULL){ /*將pb剩余結(jié)點鏈入LC*/
p = pb->next;
pb->next = LC->next;
LC->next = pb;
pb = p;
}
free(LA);
free(LB);
}
14.在一個循環(huán)鏈表中尋找值為x的結(jié)點,并將該結(jié)點移到第一個結(jié)點之前。
【解答】設此循環(huán)鏈表為單鏈表,則其類型定義與一般單鏈表相同
typedef struct mynode cyclelisttype;
void search_movefirst(cyclelisttype *CL, int x){
cyclelisttype * p , *pre;
if(CL == NULL) return;
p = CL->next;
pre = CL;
while(p != CL)
/*查找x,注意與普通單鏈表在判斷是否到表尾上的不同*/
if(p->data == x)
break;
else{
pre = p;
p = p->next;
}
if(p != CL){ /*查找成功*/
per->next = p->next; /*在原來位置刪除p*/
p->next = CL->next; /*p插入到CL后*/
CL->next = p;
}
16.鐵路進行列車調(diào)度時, 常把站臺設計成棧式結(jié)構(gòu)的站臺,如右圖所示。試問:
(1) 設有編號為1,2,3,4,5,6的六輛列車, 順序開入棧式結(jié)構(gòu)的站臺, 則可能的出棧序列有多少種?
(2) 若進站的六輛列車順序如上所述, 那么是否能夠得到435612, 325641, 154623和135426的出站序列, 如果不能, 說明為什么不能; 如果能, 說明如何得到(即寫出"進棧"或"出棧"的序列)。
【解答】
(1) 可能的不同出棧序列有 種。
(2) 不能得到435612和154623這樣的出棧序列。因為若在4, 3, 5, 6之后再將1, 2出棧,則1, 2必須一直在棧中,此時1先進棧,2后進棧,2應壓在1上面,不可能1先于2出棧。154623也是這種情況。出棧序列325641和135426可以得到。
3
5
6
2
2
4
4
4
4
1
1
1
1
1
1
1
1
3 32 32 325 325 3256 32564 325641
5
3
4
4
1
2
2
2
2
6
1 1 13 135 1354 13542 13542 135426
17.試證明:若借助棧可由輸入序列1, 2, 3, …, n得到一個輸出序列p1, p2, p3, …, pn (它是輸入序列的某一種排列),則在輸出序列中不可能出現(xiàn)以下情況,即存在i < j < k,使得pj < pk < pi。(提示:用反證法)
【解答】
因為借助棧由輸入序列1, 2, 3, …, n,可得到輸出序列p1, p2, p3, …, pn ,如果存在下標i, j, k,滿足i < j < k,那么在輸出序列中,可能出現(xiàn)如下5種情況:
① i進棧,i出棧,j進棧,j出棧,k進棧,k出棧。此時具有最小值的排在最前面pi位置,具有中間值的排在其后pj位置,具有最大值的排在pk位置,有pi < pj < pk, 不可能出現(xiàn)pj < pk < pi的情形;
② i進棧,i出棧,j進棧,k進棧,k出棧,j出棧。此時具有最小值的排在最前面pi位置,具有最大值的排在pj位置,具有中間值的排在最后pk位置,有pi < pk < pj , 不可能出現(xiàn)pj < pk < pi的情形;
③ i進棧,j進棧,j出棧,i出棧,k進棧,k出棧。此時具有中間值的排在最前面pi位置,具有最小值的排在其后pj位置,有pj < pi < pk, 不可能出現(xiàn)pj < pk < pi的情形;
④ i進棧,j進棧,j出棧,k進棧,k出棧,i出棧。此時具有中間值的排在最前面pi 位置,具有最大值的排在其后pj位置,具有最小值的排在pk位置,有pk < pi < pj, 也不可能出現(xiàn)pj < pk < pi的情形;
⑤ i進棧,j進棧,k進棧,k出棧,j出棧,i出棧。此時具有最大值的排在最前面pi 位置,具有中間值的排在其后pj位置,具有最小值的排在pk位置,有pk < pj < pi, 也不可能出現(xiàn)pj < pk < pi的情形;
18.將編號為0和1的兩個棧存放于一個數(shù)組空間V[m]中,棧底分別處于數(shù)組的兩端。當?shù)?號棧的棧頂指針top[0]等于-1時該棧為空,當?shù)?號棧的棧頂指針top[1]等于m時該棧為空。兩個棧均從兩端向中間增長。當向第0號棧插入一個新元素時,使top[0]增1得到新的棧頂位置,當向第1號棧插入一個新元素時,使top[1]減1得到新的棧頂位置。當top[0]+1 == top[1]時或top[0] == top[1]-1時,棧空間滿,此時不能再向任一棧加入新的元素。試定義這種雙棧(Double Stack)結(jié)構(gòu)的類定義,并實現(xiàn)判??铡⑴袟M、插入、刪除算法。
【解答】
bot[0] top[0] top[1] bot[1]
雙棧的類型定義如下:
typedef struct{
int top0,top1;
elemtype data[MAXSIZE];
}DoubleStack;
判棧空
int DoubleStackEmpty(DoubleStack * DS,int StackNo){
/*DS:指向雙棧的指針,StackNo:0或1,指定要判斷的是第0棧,還是第一棧
if(StackNo==0)
return(DS->top0 < 0);
else
retrun(DS->top1 >= MAXSIZE);
}
判棧滿
int DoubleStackFull(DoubleStack * DS){
return(DS->top1-DS->top0==1);
}
入棧
int PopDoubleStack(DoubleStack * DS,int StackNo,elemtype x){
/*將數(shù)據(jù)元素x插入到棧StackNo中*/
if(DoubleStackFull(DS)) return(FALSE);/*棧滿溢出*/
if(StackNo==0){/*對第0棧插入*/
DS->top0++;
DS->data[top0]=x;
}
else{/*對第1棧插入*/
DS->top1--;
DS->data[top1]=x;
}
return(TRUE);
}
刪除算法
elemtype PushDoubleStack(DoubleStack * DS,int StackNo){
/*從StackNo棧中刪除并返回一個元素,若??眨瑒t返回NULL*/
if(DoubleStackEmpty(DS,StackNo)) return(NULL);
if(StackNo==0){/*對0號棧進行操作*/
DS->top0--;
return(DS->data[DS->top0+1]);
}
else{
DS->top1++;
return(DS->data[DS->top1-1]);
}
}
19.改寫順序棧的進棧成員函數(shù)Push(x),要求當棧滿時執(zhí)行一個stackFull( )操作進行棧滿處理。其功能是:動態(tài)創(chuàng)建一個比原來的棧數(shù)組大二倍的新數(shù)組,代替原來的棧數(shù)組,原來棧數(shù)組中的元素占據(jù)新數(shù)組的前MaxSize位置。
【解答】
void push(stack * S,elemtype x) {
if(StackEmpty(S)) stackFull(S); //棧滿,做溢出處理
S->top ++;
S->data[S->top] = x; //進棧
}
void stackFull(stack *S){
elemtype *temp;
temp=calloc(3*MAXSIZE,sizeof(elemtype)); //創(chuàng)建體積大二倍的數(shù)組
for ( int i = 0; i <= S->top; i++ ) //傳送原數(shù)組的數(shù)據(jù)
temp[i] = S->data[i];
free(S->data); //刪去原數(shù)組
MAXSIZE *= 3; //數(shù)組最大體積增長二倍
S->data = temp; //新數(shù)組成為棧的數(shù)組空間
}
20.根據(jù)教案中給出的優(yōu)先級,回答以下問題:
(1) 在表達式求值的過程中,如果表達式e含有n個運算符和分界符,問操作數(shù)棧(NS)中最多可存入多少個元素?
(2) 如果表達式e含有n個運算符,且括號嵌套的最大深度為6層,問棧中最多可存入多少個元素?
【解答】
(1) 如果表達式e含有n個運算符和分界符,則可能的運算對象有n+1個。因此在利用后綴表達式求值時所用到的運算對象棧中最多可存入n+1個元素。
(2) 同上。
21.表達式的中綴表示為a * x - b / x^2,試利用棧將它改為后綴表示ax * bx2^/ -。寫出轉(zhuǎn)換過程中棧的變化。(^表示乘方運算)
【解答】
若設當前掃描到的運算符ch的優(yōu)先級為icp(ch),該運算符進棧后的優(yōu)先級為isp(ch),則可規(guī)定各個算術(shù)運算符的優(yōu)先級如下表所示。
運算符
;
(
^
*,/, %
+, -
)
isp
0
1
7
5
3
8
icp
0
8
6
4
2
1
isp也叫做棧內(nèi)(in stack priority)優(yōu)先數(shù),icp也叫做棧外(in coming priority)優(yōu)先數(shù)。當剛掃描到的運算符ch的icp(ch)大于isp(stack)時,則ch進棧;當剛掃描到的運算符ch的icp(ch)小于isp(stack)時,則位于棧頂?shù)倪\算符退棧并輸出。從表中可知,icp(“(”)最高,但當“(”進棧后,isp(“(”)變得極低。其它運算符進入棧中后優(yōu)先數(shù)都升1,這樣可體現(xiàn)在中綴表達式中相同優(yōu)先級的運算符自左向右計算的要求。運算符優(yōu)先數(shù)相等的情況只出現(xiàn)在括號配對“)”或棧底的“;”號與輸入流最后的“;”號配對時。前者將連續(xù)退出位于棧頂?shù)倪\算符,直到遇到“(”為止。然后將“(”退棧以對消括號,后者將結(jié)束算法。
步序
掃描項
項類型
動 作
棧的變化
輸 出
0
F ';' 進棧, 讀下一符號
;
1
a
操作數(shù)
F 直接輸出, 讀下一符號
;
A
2
*
操作符
F isp ( ' ; ' ) < icp ( ' * ' ), 進棧, 讀下一符號
; *
A
3
x
操作數(shù)
F 直接輸出, 讀下一符號
; *
a x
4
-
操作符
F isp ( ' * ' ) > icp ( ' - ' ), 退棧輸出
;
a x *
F isp ( ' ; ' ) < icp ( ' - ' ), 進棧, 讀下一符號
; -
a x *
5
b
操作數(shù)
F 直接輸出, 讀下一符號
; -
a x * b
6
/
操作符
F isp ( ' - ' ) < icp ( '/' ), 進棧, 讀下一符號
; -/
a x * b
7
x
操作數(shù)
F 直接輸出, 讀下一符號
; -/
a x * b x
8
^
操作符
F isp ( ' / ' ) < icp ( '^' ), 進棧, 讀下一符號
; -/^
a x * b x
9
2
操作數(shù)
F 直接輸出, 讀下一符號
; -/^
a x * b x 2
10
;
操作符
F isp ( '↑' ) > icp ( ' ; ' ), 退棧輸出
; -/
a x * b x 2^
F isp ( ' / ' ) > icp ( ' ; ' ), 退棧輸出
; -
a x * b x 2^/
F isp ( ' - ' ) > icp ( ' ; ' ), 退棧輸出
;
a x * b x 2^/ -
F 結(jié)束
22.設循環(huán)隊列的容量為70(序號從1到70),經(jīng)過一系列入隊和退隊運算后,有:
(1)front=15,rear=46;
(2)front=45,rear=10
問在上述兩種情況下,循環(huán)隊列中各有多少個元素?
【解答】
(1) 隊列中元素的個數(shù)為(MAXSIZE+rear-front) %MAXSIZE = (70+46-15)%70=31
(2) 隊列中元素的個數(shù)為(MAXSIZE+rear-front) %MAXSIZE = (70+10-45)%70=35
23.設用鏈表表示一個雙端隊列,要求可在表的兩端插入,但限制只能在表的一端刪除。試編寫基于此結(jié)構(gòu)的隊列的插入(enqueue)和刪除(dequeue)算法,并給出隊列空和隊列滿的條件。
設此隊列含頭結(jié)點,隊頭指針front指向頭結(jié)點,限定在表頭端刪除
隊列空:隊尾指針指向頭結(jié)點,即rear=front時,隊空
隊列滿:因為是鏈表,所以,只有內(nèi)存空間不足時才會有隊滿的問題。
void insertqueue(queuetype *Q,int location,elemtype x){
/*Q:指向隊列的指針;location:插入位置,0表示隊頭,1表示隊尾;x:數(shù)據(jù)元素*/
node *p;
p=(node *)malloc(sizeof(node));
p->data = x;
if(location==0){/*在隊頭處插入*/
p->next = Q->front->next;
Q->front->next = p;
if(Q->rear==Q->front) Q->rear = p;/*對空表插入*/
}
else{/*在隊尾處插入*/
p->next = NULL;
Q->rear->next = p;
Q->rear = p;
}
}
elemtype deletequeue(queuetype *Q){
elemtype x;
node * p;
if(Q->rear == Q->front) return(NULL); /*空隊列*/
p = Q->front->next;
x = p->data;
Q->front->next = p->next;
if(Q->rear == p) Q->rear = Q->front /*對只有一個元素的隊列進行刪除*/ free(p);
return(x);
}
24.所謂回文,是指從前向后順讀和從后向前倒讀都一樣的不含空白字符的串。例如did,madamimadam,pop即是回文。試編寫一個算法,以判斷一個串是否是回文。
【解答1】
將字符串中全部字符進棧,然后將棧中的字符逐個與原字符串中的字符進行比較。算法如下:
int palindrome (string S) {
stacktype * st;
int yes = 1, i = 0;
while(S[i] != “\0”){
push (st,S[i]); i++; /*掃描字符串,所有字符進棧*/
}
i = 0;
while(S[i] != “\0” ) /*比較字符串*/
if(S[i] == st.GetTop( )){pop(st); i++; }
else {yes = 0; break; }
return(yes);
}
【解答2】
采用遞歸算法,判斷從s到e中的字符串是否回文,通過函數(shù)返回是或不是。
int palindrome(char A[ ], int start, int end) {
if (A[start] != A[end] ) return 0;
else if ( s > e ) return 1;
else palindrome ( A, start+1, end-1 );
}
25.設A[n][n]為一個上三角矩陣,將A中所有非零元素按列序順次存入一維數(shù)組B中,則A與B元素的對應關系是什么?
【解答】上三角矩陣如下所示
其數(shù)據(jù)元素滿足Aij=0,對所有的i>j
上三角矩陣正好是一個下三角矩陣的轉(zhuǎn)秩,我們知道,一個下三角矩陣按行壓縮存儲時,矩陣元素與一維數(shù)組的對應關系為
A[i][j]àB[i*(i+1)/2+j],對所有的i>j;
因此,下三角矩陣按列序存儲在一維數(shù)組中的元素對應關系為
A[i][j]àB[j*(j+1)/2+i],對所有的i
下載提示(請認真閱讀)
- 1.請仔細閱讀文檔,確保文檔完整性,對于不預覽、不比對內(nèi)容而直接下載帶來的問題本站不予受理。
- 2.下載的文檔,不會出現(xiàn)我們的網(wǎng)址水印。
- 3、該文檔所得收入(下載+內(nèi)容+預覽)歸上傳者、原創(chuàng)作者;如果您是本文檔原作者,請點此認領!既往收益都歸您。
文檔包含非法信息?點此舉報后獲取現(xiàn)金獎勵!
下載文檔到電腦,查找使用更方便
32
積分
- 配套講稿:
如PPT文件的首頁顯示word圖標,表示該PPT已包含配套word講稿。雙擊word圖標可打開word文檔。
- 特殊限制:
部分文檔作品中含有的國旗、國徽等圖片,僅作為作品整體效果示例展示,禁止商用。設計者僅對作品中獨創(chuàng)性部分享有著作權(quán)。
- 關 鍵 詞:
-
計算機軟件技術(shù)基礎
習題一解答
計算機
軟件技術(shù)
基礎
習題
解答
- 溫馨提示:
1: 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
2: 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
3.本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會有圖紙預覽,若沒有圖紙預覽就沒有圖紙。
4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
5. 裝配圖網(wǎng)僅提供信息存儲空間,僅對用戶上傳內(nèi)容的表現(xiàn)方式做保護處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負責。
6. 下載文件中如有侵權(quán)或不適當內(nèi)容,請與我們聯(lián)系,我們立即糾正。
7. 本站不保證下載資源的準確性、安全性和完整性, 同時也不承擔用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。
裝配圖網(wǎng)所有資源均是用戶自行上傳分享,僅供網(wǎng)友學習交流,未經(jīng)上傳用戶書面授權(quán),請勿作他用。
鏈接地址:http://zhongcaozhi.com.cn/p-1584476.html