《數(shù)據(jù)結(jié)構(gòu)嚴蔚敏》PPT課件.ppt
《《數(shù)據(jù)結(jié)構(gòu)嚴蔚敏》PPT課件.ppt》由會員分享,可在線閱讀,更多相關(guān)《《數(shù)據(jù)結(jié)構(gòu)嚴蔚敏》PPT課件.ppt(708頁珍藏版)》請在裝配圖網(wǎng)上搜索。
1、第 3章 棧和隊列 棧和隊列是兩種應(yīng)用非常廣泛的數(shù)據(jù)結(jié)構(gòu) ,它們都 來自線性表數(shù)據(jù)結(jié)構(gòu), 都是 “ 操作受限 ” 的線性表 。 棧在計算機的實現(xiàn)有多種方式: 硬堆棧 :利用 CPU中的某些寄存器組或類似的硬 件或使用內(nèi)存的特殊區(qū)域來實現(xiàn)。這類堆棧容量有限, 但速度很快; 軟堆棧 :這類堆棧主要在內(nèi)存中實現(xiàn)。堆棧容量 可以達到很大。在實現(xiàn)方式上,又有 動態(tài)方式 和 靜態(tài) 方式 兩種。 本章將討論棧和隊列的基本概念 、 存儲結(jié)構(gòu) 、 基本 操作以及這些操作的具體實現(xiàn) 。 3.1 棧 3.1.1 棧的基本概念 1 棧的概念 棧 (Stack): 是限制在表的一端進行插入和刪除操 作的線性表。又稱為后
2、進先出 LIFO (Last In First Out) 或先進后出 FILO (First In Last Out)線性 表。 棧頂 (Top): 允許進行插入、刪除操作的一端,又 稱為表尾。用棧頂指針 (top)來指示棧頂元素 。 棧底 (Bottom): 是固定端,又稱為表頭。 空棧 : 當表中沒有元素時稱為空棧。 設(shè)棧 S=(a1, a2, a n),則 a1稱 為棧底元素, an為棧頂元素,如圖 3- 1所示。 棧中元素按 a1, a2, a n的次序 進棧,退棧的第一個元素應(yīng)為棧頂元 素。即棧的修改是按后進先出的原則 進行的。 圖 3-1 順序 棧示意圖 a1 a2 ai an b
3、ottom top 進棧( push) 出棧 (pop) 2 棧的抽象數(shù)據(jù)類型定義 ADT Stack 數(shù)據(jù)對象: D = ai|ai ElemSet, i=1,2,n , n0 數(shù)據(jù)關(guān)系: R =|ai-1, ai D, i=2,3,n 基本操作:初始化、進棧、出棧、取棧頂元素等 ADT Stack 棧的順序存儲結(jié)構(gòu)簡稱為順序棧,和線性表相類似, 用 一維數(shù)組 來 存儲棧 。根據(jù)數(shù)組是否可以根據(jù)需要增大, 又可分為 靜態(tài)順序棧 和 動態(tài)順序棧 。 靜態(tài)順序棧 實現(xiàn)簡單,但不能根據(jù)需要增大棧的 存儲空間; 動態(tài)順序棧 可以根據(jù)需要增大棧的存儲空間,但 實現(xiàn)稍為復(fù)雜。 3.1.2 棧的順序存儲表
4、示 采用 動態(tài) 一維數(shù)組 來 存儲棧 。所謂動態(tài),指的是棧 的大小可以根據(jù)需要增加。 用 bottom表示棧底指針,棧底固定不變的;棧頂 則隨著進棧和退棧操作而變化。用 top(稱為棧頂指針 ) 指示當前棧頂位置。 用 top=bottom作為??盏臉擞?,每次 top指向棧頂 數(shù)組中的下一個存儲位置 。 結(jié)點進棧 : 首先將數(shù)據(jù)元素保存到棧頂 (top所指 的當前位置 ),然后執(zhí)行 top加 1,使 top指向棧頂?shù)南?一個存儲位置 ; 3.1.2.1 棧的動態(tài)順序存儲表示 結(jié)點出棧 : 首先執(zhí)行 top減 1,使 top指向棧頂元 素的存儲位置,然后將棧頂元素取出 。 圖 3-2是一個動態(tài)
5、棧的變化示意圖 。 圖 3-2 (動態(tài) )堆棧變化示意圖 空棧 bottom top 元素 a進棧 bottom top a 元素 b, c進棧 bottom top a b c 元素 c退棧 bottom top a b bottom top a b d e f 元素 d, e, f進棧 基本操作的實現(xiàn) 1 棧的類型定義 #define STACK_SIZE 100 /* 棧初始向量大小 */ #define STACKINCREMENT 10 /* 存儲空間分配增量 */ #typedef int ElemType ; typedef struct sqstack ElemType *bo
6、ttom; /* 棧不存在時值為 NULL */ ElemType *top; /* 棧頂指針 */ int stacksize ; /* 當前已分配空間,以元素為單位 */ SqStack ; 2 棧的初始化 Status Init_Stack(void) SqStack S ; S.bottom=(ElemType *)malloc(STACK_SIZE *sizeof(ElemType); if (! S.bottom) return ERROR; S.top=S.bottom ; /* 棧空時棧頂和棧底指針相同 */ S. stacksize=STACK_SIZE; return OK
7、 ; 3 壓棧 (元素進棧 ) Status push(SqStack S , ElemType e) if (S.top-S.bottom=S. stacksize-1) S.bottom=(ElemType *)realloc(S. STACKINCREMENT+STACK_SIZE) *sizeof(ElemType); /* 棧滿,追加存儲空間 */ if (! S.bottom) return ERROR; S.top=S.bottom+S. stacksize ; S. stacksize+=STACKINCREMENT ; *S.top=e; S.top+ ; /* 棧頂指針加
8、1, e成為新的棧頂 */ return OK; 4 彈棧 (元素 出棧 ) Status pop( SqStack S, ElemType *e ) /*彈出棧頂元素 */ if ( S.top= S.bottom ) return ERROR ; /* 棧空,返回失敗標志 */ S.top- ; e=*S. top ; return OK ; 采用 靜態(tài) 一維數(shù)組 來 存儲棧 。 棧底固定不變的,而棧頂則隨著進棧和退棧操作變 化的, 棧底固定不變的;棧頂則隨著進棧和退棧操作而 變化,用一個整型變量 top(稱為棧頂指針 )來指示當 前棧頂位置。 用 top=0表示??盏某跏紶顟B(tài) ,每次 t
9、op指向棧頂 在數(shù)組中的存儲位置 。 結(jié)點進棧 : 首先執(zhí)行 top加 1,使 top指向新的棧 頂位置 , 然后將數(shù)據(jù)元素保存到棧頂 (top所指的當前 位置 )。 3.1.2.2 棧的靜態(tài)順序存儲表示 結(jié)點出棧 : 首先 把 top指向的棧頂元素取出 , 然 后執(zhí)行 top減 1,使 top指向新的棧頂位置 。 若棧的數(shù)組有 Maxsize個元素 ,則 top=Maxsize-1時 棧滿 。圖 3-3是一個大小為 5的棧的變化示意圖 。 圖 3-3 靜態(tài) 堆棧變化示意圖 空棧 bottom top Top=1 1個元素進棧 bottom top a Top=3 3個元素進棧 bottom
10、top a b c Top=4 棧滿 bottom top a b e d Top=2 元素 c進棧 bottom top a b 基本操作的實現(xiàn) 1 棧的類型定義 # define MAX_STACK_SIZE 100 /* 棧向量大小 */ # typedef int ElemType ; typedef struct sqstack ElemType stack_arrayMAX_STACK_SIZE ; int top; SqStack ; 2 棧的初始化 SqStack Init_Stack(void) SqStack S ; S.bottom=S.top=0 ; return(S)
11、 ; 3 壓棧 (元素進棧 ) Status push(SqStack S , ElemType e) /* 使數(shù)據(jù)元素 e進棧成為新的棧頂 */ if (S.top=MAX_STACK_SIZE-1) return ERROR; /* 棧滿,返回錯誤標志 */ S.top+ ; /* 棧頂指針加 1 */ S.stack_arrayS.top=e ; /* e成為新的棧頂 */ return OK; /* 壓棧成功 */ 4 彈棧 (元素 出棧 ) Status pop( SqStack S, ElemType *e ) /*彈出棧頂元素 */ if ( S.top=0 ) return E
12、RROR ; /* ??眨祷劐e誤標志 */ *e=S.stack_arrayS.top ; S.top- ; return OK ; 當棧滿時做進棧運算必定產(chǎn)生空間溢出,簡稱 “ 上 溢 ” 。上溢是一種出錯狀態(tài),應(yīng)設(shè)法避免。 當??諘r做退棧運算也將產(chǎn)生溢出,簡稱 “ 下溢 ” 。 下溢則可能是正常現(xiàn)象,因為棧在使用時,其初態(tài)或終 態(tài)都是空棧,所以下溢常用來作為控制轉(zhuǎn)移的條件。 1 棧的鏈式表示 棧的 鏈式存儲結(jié)構(gòu) 稱為鏈棧,是運算受限的單鏈表。 其插入和刪除操作只能在表頭位置上進行。因此,鏈棧 沒有必要像單鏈表那樣附加頭結(jié)點,棧頂指針 top就是 鏈表的頭指針。圖 3-4是棧的鏈式存儲表示
13、形式 。 3.1.3 棧的鏈式存儲表示 空棧 top 非空棧 top a4 a3 a1 a2 圖 3-4 鏈 棧存儲形式 鏈棧的 結(jié)點類型說明如下: typedef struct Stack_Node ElemType data ; struct Stack_Node *next ; Stack_Node ; 2 鏈?;静僮鞯膶崿F(xiàn) (1) 棧的初始化 Stack_Node *Init_Link_Stack(void) Stack_Node *top ; top=(Stack_Node *)malloc(sizeof(Stack_Node ) ; top-next=NULL ; return(
14、top) ; (2) 壓棧 (元素進棧 ) Status push(Stack_Node *top , ElemType e) Stack_Node *p ; p=(Stack_Node *)malloc(sizeof(Stack_Node) ; if (!p) return ERROR; /* 申請新結(jié)點失敗,返回錯誤標志 */ p-data=e ; p-next=top-next ; top-next=p ; /* 鉤鏈 */ return OK; (3) 彈棧 (元素 出棧 ) Status pop(Stack_Node *top , ElemType *e) /* 將棧頂元素 出 棧
15、*/ Stack_Node *p ; ElemType e ; if (top-next=NULL ) return ERROR ; /* 棧空,返回錯誤標志 */ p=top-next ; e=p-data ; /* 取棧頂元素 */ top-next=p-next ; /* 修改棧頂指針 */ free(p) ; return OK ; 3.2 棧的應(yīng)用 由于棧具有的 “ 后進先出 ” 的固有特性,因此,棧 成為程序設(shè)計中常用的工具和數(shù)據(jù)結(jié)構(gòu)。以下是幾個棧 應(yīng)用的例子。 3.2.1 數(shù)制轉(zhuǎn)換 十進制整數(shù) N向其它進制數(shù) d(二 、 八 、 十六 )的轉(zhuǎn)換 是計算機實現(xiàn)計算的基本問題。 轉(zhuǎn)換
16、法則 : 該轉(zhuǎn)換法則對應(yīng)于一個簡單算法原理 : n=(n div d)*d+n mod d 其中: div為整除運算 ,mod為求余運算 例如 (1348)10= (2504)8, 其運算過程如下: n n div 8 n mod 8 1348 168 4 168 21 0 21 2 5 2 0 2 采用靜態(tài)順序棧方式實現(xiàn) void conversion(int n , int d) /*將 十進制整數(shù) N轉(zhuǎn)換為 d(2或 8)進制數(shù) */ SqStack S ; int k, *e ; S=Init_Stack(); while (n0) k=n%d ; push(S , k) ; n=n/
17、d ; /* 求出所有的余 數(shù), 進棧 */ while (S.top!=0) /* 棧不空時出棧,輸出 */ pop(S, e) ; printf(%1d , *e) ; 3.2.2 括號匹配問題 在文字處理軟件或編譯程序設(shè)計時,常常需要檢查 一個字符串或一個表達式中的括號是否相匹配 ? 匹配思想 : 從左至右掃描一個字符串 (或表達式 ),則 每個右括號將與最近遇到的那個左括號相匹配 。則可以 在從左至右掃描過程中把所遇到的左括號存放到堆棧中。 每當遇到一個右括號時,就將它與棧頂?shù)淖罄ㄌ?(如果 存在 )相匹配,同時從棧頂刪除該左括號。 算法思想 : 設(shè)置一個棧,當讀到左括號時,左括號進
18、棧。當讀到右括號時,則從棧中彈出一個元素,與讀到 的左括號進行匹配,若匹配成功,繼續(xù)讀入;否則匹配 失敗,返回 FLASE。 算法描述 #define TRUE 0 #define FLASE -1 SqStack S ; S=Init_Stack() ; /*堆棧初始化 */ int Match_Brackets( ) char ch , x ; scanf(%c , while (asc(ch)!=13) if (ch=()|(ch=) push(S , ch) ; else if (ch=) x=pop(S) ; if (x!=) printf(括號不匹配” ) ; return FLA
19、SE ; else if (ch=) x=pop(S) ; if (x!=() printf(括號不匹配” ) ; return FLASE ; if (S.top!=0) printf(括號數(shù)量不匹配!” ) ; return FLASE ; else return TRUE ; 3.2.2 棧與遞歸調(diào)用的實現(xiàn) 棧的另一個重要應(yīng)用是在程序設(shè)計語言中實現(xiàn)遞歸 調(diào)用。 遞歸調(diào)用 : 一個函數(shù) (或過程 )直接或間接地調(diào)用 自己本身,簡稱 遞歸 (Recursive)。 遞歸 是程序設(shè)計中的一個強有力的工具。因為遞歸 函數(shù)結(jié)構(gòu)清晰,程序易讀,正確性很容易得到證明。 為了使遞歸調(diào)用不至于無終止地進行
20、下去,實際上 有效的遞歸調(diào)用函數(shù) (或過程 )應(yīng)包括兩部分: 遞推規(guī)則 (方法 ), 終止條件 。 例如:求 n! Fact(n)= 1 當 n=0時 終止條件 n*fact(n-1) 當 n0時 遞推規(guī)則 為保證遞歸調(diào)用正確執(zhí)行,系統(tǒng)設(shè)立一個 “ 遞歸工 作棧 ” ,作為整個遞歸調(diào)用過程期間使用的數(shù)據(jù)存儲區(qū)。 每一層遞歸包含的信息如: 參數(shù) 、 局部變量 、 上一 層的返回地址構(gòu)成 一個“ 工作記錄 ” 。每進入一層遞 歸,就產(chǎn)生一個新的工作記錄壓入棧頂;每退出一層遞 歸,就從棧頂彈出一個工作記錄。 從被調(diào)函數(shù)返回調(diào)用函數(shù)的一般步驟: (1) 若棧為空,則執(zhí)行正常返回。 從棧頂彈出一個工作記
21、錄。 將“工作記錄”中的參數(shù)值、局部變量值賦給相 應(yīng)的變量;讀取返回地址。 將函數(shù)值賦給相應(yīng)的變量。 (5) 轉(zhuǎn)移到返回地址。 1 隊列的基本概念 隊列 (Queue):也是運算受限的線性表。是一種 先 進先出 (First In First Out ,簡稱 FIFO)的線性表。只允 許在表的一端進行插入,而在另一端進行刪除。 隊首 (front) :允許進行刪除的一端稱為隊首。 隊尾 (rear) :允許進行插入的一端稱為隊尾。 例如:排隊購物。操作系統(tǒng)中的作業(yè)排隊。先進入 隊列的成員總是先離開隊列。 3.3 隊 列 3.3.1 隊列及其基本概念 隊列中沒有元素時稱為空隊列。在空隊列中依次
22、加入元素 a1, a2, , a n之后, a1是隊首元素, an是隊尾元 素。顯然退出隊列的次序也只能是 a1, a2, , a n ,即隊 列的修改是依 先進先出 的原則進行的,如圖 3-5所示。 a1 , a2 , , a n 出隊 入隊 隊尾 隊首 圖 3-5 隊列示意圖 2 隊列的抽象數(shù)據(jù)類型定義 ADT Queue 數(shù)據(jù)對象: D = ai|ai ElemSet, i=1, 2, , n, n = 0 數(shù)據(jù)關(guān)系: R = | ai-1, ai D, i=2,3,n 約定 a1端為隊首, an端為隊尾。 基本操作: Create():創(chuàng)建一個空隊列 ; EmptyQue():若隊列為
23、空 , 則返回 true , 否則返 回 flase ; InsertQue(x) :向隊尾插入元素 x; DeleteQue(x) :刪除隊首元素 x; ADT Queue 3.3.2 隊列的順序表示和實現(xiàn) 利用 一組連續(xù)的存儲單元 (一維數(shù)組 ) 依次存放從隊 首到隊尾的各個元素,稱為 順序隊列 。 對于隊列,和順序棧相類似,也有動態(tài)和靜態(tài)之分。 本部分介紹的是 靜態(tài)順序隊列 ,其類型定義如下: #define MAX_QUEUE_SIZE 100 typedef struct queue ElemType Queue_arrayMAX_QUEUE_SIZE ; int front ; i
24、nt rear ; SqQueue; 3.3.2.1 隊列的順序 存儲結(jié)構(gòu) 設(shè)立一個 隊首指針 front ,一個 隊尾指針 rear ,分 別指向隊首和隊尾元素 。 初始化 : front=rear=0。 入隊 : 將新元素插入 rear所指的位置,然后 rear加 1。 出隊 : 刪去 front所指的元素,然后加 1并返回被刪 元素。 隊列為空 : front=rear。 隊滿 : rear=MAX_QUEUE_SIZE-1或 front=rear。 在非空隊列里,隊首指針始終指向隊頭元素,而隊 尾指針始終指向隊尾元素的下一位置。 順序隊列中存在 “ 假溢出 ” 現(xiàn)象。因為在入隊和出 隊
25、操作中,頭、尾指針只增加不減小,致使被刪除元素 的空間永遠無法重新利用。因此,盡管隊列中實際元素 個數(shù)可能遠遠小于數(shù)組大小,但可能由于尾指針巳超出 向量空間的上界而不能做入隊操作。該現(xiàn)象稱為 假溢出 。 如圖 3-6所示是數(shù)組大小為 5的順序隊列中隊首 、 隊尾指 針和隊列中元素的變化情況。 (a) 空隊列 Q.front Q.rear (b) 入隊 3個元素 a3 a2 a1 Q.front Q.rear (c) 出隊 3個元素 Q.front Q.rear (d) 入隊 2個元素 a5 a4 Q.front Q.rear 圖 3-6 隊列示意圖 3.3.2.2 循環(huán)隊列 為充分利用向量空間
26、,克服上述“ 假溢出 ”現(xiàn)象的 方法是:將為隊列分配的 向量空間看成為一個首尾相接 的圓環(huán) ,并稱這種隊列為 循環(huán)隊列 (Circular Queue)。 在循環(huán)隊列中進行出隊、入隊操作時,隊首、隊尾 指針仍要加 1,朝前移動。只不過當隊首、隊尾指針指 向向量上界 (MAX_QUEUE_SIZE-1)時,其加 1操作的 結(jié)果是指向向量的下界 0。 這種循環(huán)意義下的加 1操作可以描述為: if (i+1=MAX_QUEUE_SIZE) i=0; else i+ ; 其中: i代表隊首指針 (front)或隊尾指針 (rear) 用模運算可簡化為: i=(i+1)%MAX_QUEUE_SIZE ;
27、 顯然,為循環(huán)隊列所分配的空間可以被充分利用, 除非向量空間真的被隊列元素全部占用,否則不會上溢 。因此,真正實用的順序隊列是循環(huán)隊列。 例:設(shè) 有循環(huán)隊列 QU0, 5,其初始狀態(tài)是 front=rear=0,各種操作后隊列的頭、尾指針的狀態(tài)變 化情況如下圖 3-7所示。 1 2 3 4 5 0 (a) 空隊列 front rear 1 2 3 4 5 0 (b) d, e, b, g入 隊 front d e b g rear 1 2 3 4 5 0 (c) d, e出隊 b g front rear 入隊時尾指針向前追趕頭指針,出隊時頭指針向前 追趕尾指針,故隊空和隊滿時頭尾指針均相等。
28、因此, 無法通過 front=rear來判斷隊列 “ 空 ” 還是 “ 滿 ” 。解 決此問題的方法是:約定入隊前,測試尾指針在循環(huán)意 義下加 1后是否等于頭指針,若相等則認為隊滿 。即 : rear所指的單元始終為空。 1 2 3 4 5 0 (d) i, j, k入 隊 b g front i j k rear 1 2 3 4 5 0 (e) b, g出隊 i j k rear front 1 2 3 4 5 0 (f) r, p, s, t入隊 i j k front r p rear 圖 3-7 循環(huán)隊列操作及指針變化情況 循環(huán)隊列為空 : front=rear 。 循環(huán)隊列滿 : (
29、rear+1)%MAX_QUEUE_SIZE =front。 循環(huán)隊列的基本操作 1 循環(huán)隊列的初始化 SqQueue Init_CirQueue(void) SqQueue Q ; Q.front=Q.rear=0; return(Q) ; 2 入隊操作 Status Insert_CirQueue(SqQueue Q , ElemType e) /* 將數(shù)據(jù)元素 e插入到循環(huán)隊列 Q的隊尾 */ if (Q.rear+1)%MAX_QUEUE_SIZE= Q.front) return ERROR; /* 隊滿,返回錯誤標志 */ Q.Queue_arrayQ.rear=e ; /* 元素
30、 e入隊 */ Q.rear=(Q.rear+1)% MAX_QUEUE_SIZE ; /* 隊尾指針向前移動 */ return OK; /* 入隊成功 */ 3 出隊操作 Status Delete_CirQueue(SqQueue Q, ElemType *x ) /* 將循環(huán)隊列 Q的隊首元素出隊 */ if (Q.front+1= Q.rear) return ERROR ; /* 隊空,返回錯誤標志 */ *x=Q.Queue_arrayQ.front ; /* 取隊首元素 */ Q.front=(Q.front+1)% MAX_QUEUE_SIZE ; /* 隊首指針向前移動 *
31、/ return OK ; 1 隊列的鏈式存儲表示 隊列的鏈式存儲結(jié)構(gòu)簡稱為鏈隊列,它是限制僅 在表頭進行刪除操作和表尾進行插入操作的單鏈表。 需要兩類不同的結(jié)點 : 數(shù)據(jù)元素 結(jié)點 , 隊列的 隊首 指針和隊尾指針 的結(jié)點 ,如圖 3-8所示。 3.3.3 隊列的鏈式表示和實現(xiàn) 數(shù)據(jù)元素結(jié)點類型定義: typedef struct Qnode ElemType data ; struct Qnode *next ; QNode ; 數(shù)據(jù)元素結(jié)點 data 指針結(jié)點 front rear 圖 3-8 鏈隊列結(jié)點示意圖 指針結(jié)點類型定義: typedef struct link_queue QN
32、ode *front , *rear ; Link_Queue ; 2 鏈隊運算及指針變化 鏈隊的操作實際上是單鏈表的操作,只不過是刪 除在表頭進行,插入在表尾進行。插入、刪除時分別修 改不同的指針。鏈隊運算及指針變化如圖 3-9所示。 圖 3-9 隊列操作及指針變化 (a) 空隊列 front rear (b) x入隊 x front rear (c) y再入隊 y front rear x (d) x出隊 y x front rear 3 鏈隊列的基本操作 鏈隊列的初始化 LinkQueue *Init_LinkQueue(void) LinkQueue *Q ; QNode *p ; p
33、=(QNode *)malloc(sizeof(QNode) ; /* 開辟頭結(jié)點 */ p-next=NULL ; Q=(LinkQueue *)malloc(sizeof(LinkQueue) ; /* 開辟鏈隊的指針結(jié)點 */ Q.front=Q.rear=p ; return(Q) ; 鏈隊列的 入隊操作 在已知隊列的隊尾插入一個元素 e ,即 修改隊尾 指 針 (Q.rear)。 Status Insert_CirQueue(LinkQueue *Q , ElemType e) /* 將數(shù)據(jù)元素 e插入到鏈隊列 Q的隊尾 */ p=(QNode *)malloc(sizeof(QNo
34、de) ; if (!p) return ERROR; /* 申請新結(jié)點失敗,返回錯誤標志 */ p-data=e ; p-next=NULL ; /* 形成新結(jié)點 */ Q.rear-next=p ; Q.rear=p ; /* 新結(jié)點插入到隊尾 */ return OK; 鏈隊列的出隊操作 Status Delete_LinkQueue(LinkQueue *Q, ElemType *x) QNode *p ; if (Q.front=Q.rear) return ERROR ; /* 隊空 */ p=Q.front-next ; /* 取隊首結(jié)點 */ *x=p-data ; Q.fro
35、nt-next=p-next ; /* 修改隊首指針 */ if (p=Q.rear) Q.rear=Q.front ; /* 當隊列只有一個結(jié)點時應(yīng)防止丟失隊尾指針 */ free(p) ; return OK ; 鏈隊列的撤消 void Destroy_LinkQueue(LinkQueue *Q ) /* 將鏈隊列 Q的隊首元素出隊 */ while (Q.front!=NULL) Q.rear=Q.front-next; /* 令尾指針指向隊列的第一個結(jié)點 */ free(Q.front); /* 每次釋放一個結(jié)點 */ /* 第一次是頭結(jié)點,以后是元素結(jié)點 */ Q.ront=Q.r
36、ear; 習 題 三 1 設(shè)有一個棧,元素進棧的次序為 a, b, c。問經(jīng)過棧 操作后可以得到哪些輸出序列? 2 循環(huán)隊列的優(yōu)點是什么 ?如何判斷它的空和滿 ? 3 設(shè)有一個靜態(tài)順序隊列,向量大小為 MAX,判斷 隊列為空的條件是什么?隊列滿的條件是什么? 4 設(shè)有一個靜態(tài)循環(huán)隊列,向量大小為 MAX,判斷 隊列為空的條件是什么?隊列滿的條件是什么? 5 利用棧的基本操作, 寫一個返回棧 S中結(jié)點個數(shù)的 算法 int StackSize(SeqStack S) ,并說明 S為何不作為指 針參數(shù) 的算法 ? 6 一個雙向棧 S是在同一向量空間內(nèi)實現(xiàn)的兩個棧 , 它們的棧底分別設(shè)在向量空間的兩端
37、。試為此雙向棧設(shè) 計初始化 InitStack(S) ,入棧 Push(S,i,x),出棧 Pop(S,i,x) 算法 ,其中 i為 0或 1 ,用以表示棧號。 7 設(shè) Q0,6是一個靜態(tài)順序隊列 ,初始狀態(tài)為 front=rear=0,請畫出做完下列操作后隊列的頭尾指針 的狀態(tài)變化情況,若不能入對,請指出其元素,并說明 理由。 a, b, c, d入隊 a, b, c出隊 i , j , k , l , m入隊 d, i出隊 n, o, p, q, r入隊 8 假設(shè) Q0,5是一個循環(huán)隊列 ,初始狀態(tài)為 front=rear=0,請畫出做完下列操作后隊列的頭尾指針 的狀態(tài)變化情況,若不能入對
38、,請指出其元素,并說明 理由。 d, e, b, g, h入隊 d, e出隊 i , j , k , l , m入隊 b出隊 n, o, p, q, r入隊 第 4章 串 在非數(shù)值處理、事務(wù)處理等問題常涉及到一系列 的字符操作。計算機的硬件結(jié)構(gòu)主要是反映數(shù)值計算的 要求,因此,字符串的處理比具體數(shù)值處理復(fù)雜。本章 討論串的存儲結(jié)構(gòu)及幾種基本的處理。 4.1 串類型的定義 4.1.1 串的基本概念 串 (字符串 ):是零個或多個字符組成的有限序列。 記作: S=a1a2a3 ,其中 S是串名, ai(1 i n)是單 個, 可以是字母、數(shù)字或其它字符。 串值 :雙引號括起來的字符序列是串值。 串
39、長 :串中所包含的字符個數(shù)稱為該串的長度。 空串 (空的字符串 ):長度為零的串稱為空串,它不 包含任何字符。 空格串 (空白串 ):構(gòu)成串的所有字符都是空格的串 稱為空白串。 注意 :空串和空白串的不同,例如 “ ” 和 “” 分別表 示長度為 1的空白串和長度為 0的空串。 子串 (substring):串中任意個連續(xù)字符組成的子序 列稱為該串的子串,包含子串的串相應(yīng)地稱為主串。 子串的序號 :將子串在主串中首次出現(xiàn)時的該子串 的首字符對應(yīng)在主串中的序號,稱為子串在主串中的序 號(或位置)。 例如, 設(shè)有串 A和 B分別是: A=這是字符串”, B=是” 則 B是 A的子串, A為主串。
40、B在 A中出現(xiàn)了兩次,其中 首次出現(xiàn)所對應(yīng)的主串位置是 3。因此,稱 B在 A中的序 號為 3 。 特別地,空串是任意串的子串,任意串是其自身的 子串。 串相等 :如果兩個串的串值相等 (相同 ),稱這兩個 串相等。換言之,只有當兩個串的長度相等,且各個對 應(yīng)位置的字符都相同時才相等。 通常在程序中使用的串可分為兩種:串變量和串常 量。 串常量和整常數(shù)、實常數(shù)一樣,在程序中只能被引 用但不能不能改變其值,即只能讀不能寫。通常串常量 是由直接量來表示的,例如語句錯誤 (溢出 ” )中 “ 溢 出 ” 是直接量。 串變量和其它類型的變量一樣,其值是可以改變。 4.1.2 串的抽象數(shù)據(jù)類型定義 AD
41、T String 數(shù)據(jù)對象: D = ai|ai CharacterSet, i=1,2,n, n 0 數(shù)據(jù)關(guān)系: R = | ai-1, ai D, i=2,3,n 基本操作: StrAssign(t , chars) 初始條件: chars是一個字符串常量。 操作結(jié)果:生成一個值為 chars的串 t 。 StrConcat(s, t) 初始條件:串 s, t 已存在。 操作結(jié)果:將串 t聯(lián)結(jié)到串 s后形成新串存放到 s中。 StrLength(t) 初始條件:字符串 t已存在。 操作結(jié)果:返回串 t中的元素個數(shù),稱為串長。 SubString (s, pos, len, sub) 初始條
42、件:串 s, 已存在 , 1 pos StrLength(s)且 0 len StrLength(s) pos+1。 操作結(jié)果:用 sub返回串 s的第 pos個字符起長度為 len 的子串。 ADT String 4.2 串的存儲表示和實現(xiàn) 串是一種特殊的線性表,其存儲表示和線性表類似, 但又不完全相同。串的存儲方式取決于將要對串所進行 的操作。串在計算機中有 3種表示方式: 定長順序存儲表示 :將串定義成字符數(shù)組,利用 串名可以直接訪問串值。用這種表示方式,串的存 儲空間在編譯時確定,其大小不能改變。 堆分配存儲方式 :仍然用一組地址連續(xù)的存儲單 元來依次存儲串中的字符序列,但串的存儲空間
43、是 在程序運行時根據(jù)串的實際長度動態(tài)分配的。 塊鏈存儲方式 :是一種鏈式存儲結(jié)構(gòu)表示。 4.2.1 串的定長順序存儲表示 這種存儲結(jié)構(gòu)又稱為串的順序存儲結(jié)構(gòu)。是用一 組連續(xù)的存儲單元來存放串中的字符序列。所謂定長順 序存儲結(jié)構(gòu),是直接使用定長的字符數(shù)組來定義,數(shù)組 的上界預(yù)先確定。 定長順序存儲結(jié)構(gòu)定義為: #define MAX_STRLEN 256 typedef struct char strMAX_STRLEN ; int length; StringType ; 1 串的聯(lián)結(jié)操作 Status StrConcat ( StringType s, StringType t) /* 將串
44、 t聯(lián)結(jié)到串 s之后,結(jié)果仍然保存在 s中 */ int i, j ; if (s.length+t.length)MAX_STRLEN) Return ERROR ; /* 聯(lián)結(jié)后長度超出范圍 */ for (i=0 ; it.length ; i+) s.strs.length+i=t.stri ; /* 串 t聯(lián)結(jié)到串 s之后 */ s.length=s.length+t.length ; /* 修改聯(lián)結(jié)后的串長度 */ return OK ; 2 求子串操作 Status SubString (StringType s, int pos, int len, StringType *su
45、b) int k, j ; if (poss.length|len(s.length-pos+1) return ERROR ; /* 參數(shù)非法 */ sub-length=len-pos+1 ; /* 求得子串長度 */ for (j=0, k=pos ; kstrj=s.stri ; /* 逐個字符復(fù)制求得子串 */ return OK ; 4.2.2 串的堆分配存儲表示 實現(xiàn)方法:系統(tǒng)提供一個空間足夠大且地址連續(xù)的 存儲空間 (稱為 “ 堆 ” )供串使用??墒褂?C語言的動態(tài) 存儲分配函數(shù) malloc()和 free()來 管理。 特點是:仍然以一組地址連續(xù)的存儲空間來存儲字 符串值
46、,但其所需的存儲空間是在程序執(zhí)行過程中動態(tài) 分配,故是動態(tài)的,變長的。 串的堆式存儲結(jié)構(gòu)的類型定義 typedef struct char *ch; /* 若非空,按長度分配,否則為 NULL */ int length; /* 串的長度 */ HString ; 1 串的聯(lián)結(jié)操作 Status Hstring *StrConcat(HString *T, HString *s1, HString *s2) /* 用 T返回由 s1和 s2聯(lián)結(jié)而成的串 */ int k, j , t_len ; if (T.ch) free(T); /* 釋放舊空間 */ t_len=s1-length+s2
47、-length ; if (p=(char *)malloc(sizeof(char)*t_len)=NULL) printf(系統(tǒng)空間不夠,申請空間失敗 ! n) ; return ERROR ; for (j=0 ; jlength; j+) T-chj=s1-chj ; /* 將串 s復(fù)制到串 T中 */ for (k=s1-length, j=0 ; jlength; k+, j+) T-chj=s1-chj ; /* 將串 s2復(fù)制到串 T中 */ free(s1-ch) ; free(s2-ch) ; return OK ; 4.2.3 串的鏈式存儲表示 串的鏈式存儲結(jié)構(gòu)和線性表的
48、串的鏈式存儲結(jié)構(gòu)類 似,采用單鏈表來存儲串, 結(jié)點的構(gòu)成是: data域:存放字符, data域可存放的字符個數(shù)稱為 結(jié)點的大??; next域:存放指向下一結(jié)點的指針。 若每個結(jié)點僅存放一個字符,則結(jié)點的指針域就非 常多,造成系統(tǒng)空間浪費,為節(jié)省存儲空間,考慮串結(jié) 構(gòu)的特殊性,使每個結(jié)點存放若干個字符,這種結(jié)構(gòu)稱 為塊鏈結(jié)構(gòu)。如 圖 4-1是塊大小為 3的串的塊鏈式存儲結(jié) 構(gòu)示意圖。 串的塊鏈式存儲的類型定義包括: 塊結(jié)點的類型定義 #define BLOCK_SIZE 4 typedef struct Blstrtype char dataBLOCK_SIZE ; struct Blstrt
49、ype *next; BNODE ; a b c e p c g head 圖 4-1 串的塊鏈式存儲結(jié)構(gòu)示意圖 (2) 塊鏈串的類型定義 typedef struct BNODE head; /* 頭指針 */ int Strlen ; /* 當前長度 */ Blstring ; 在這種存儲結(jié)構(gòu)下,結(jié)點的分配總是完整的結(jié)點為 單位,因此,為使一個串能存放在整數(shù)個結(jié)點中,在串 的末尾填上不屬于串值的特殊字符,以表示串的終結(jié)。 當一個塊 (結(jié)點 )內(nèi)存放多個字符時,往往會使操作 過程變得較為復(fù)雜,如在串中插入或刪除字符操作時通 常需要在塊間移動字符。 4.3 串的模式匹配算法 模式匹配 (模范匹
50、配 ): 子串在主串中的定位稱為模 式匹配或串匹配 (字符串匹配 ) 。模式匹配成功是指在 主串 S中能夠找到模式串 T,否則,稱模式串 T在主串 S 中不存在。 模式匹配的應(yīng)用在非常廣泛。例如,在文本編輯 程序中,我們經(jīng)常要查找某一特定單詞在文本中出現(xiàn)的 位置。顯然,解此問題的有效算法能極大地提高文本編 輯程序的響應(yīng)性能。 模式匹配是一個較為復(fù)雜的串操作過程。迄今為止, 人們對串的模式匹配提出了許多思想和效率各不相同的 計算機算法。介紹兩種主要的模式匹配算法。 4.3.1 Brute-Force模式匹配算法 設(shè) S為目標串, T為模式串,且不妨設(shè): S=s0s1s2s n-1 , T=t0t
51、1t2 t m-1 串的匹配實際上是對合法的位置 0 i n-m依次將 目標串中的子串 sii+m -1和模式串 t0m -1進行比較: 若 sii+m -1=t0m -1:則稱從位置 i開始的匹 配成功,亦稱模式 t在目標 s中出現(xiàn); 若 sii+m -1t0m -1:從 i開始的匹配失敗。 位置 i稱為位移,當 sii+m -1=t0m -1時, i稱為 有效位移;當 sii+m -1 t0m -1時, i稱為無效 位移。 這樣,串匹配 問題可簡化為找出某給定模式 T在給 定目標串 S中首次出現(xiàn) 的有效位移。 算法實現(xiàn) int IndexString(StringType s , Stri
52、ngType t , int pos ) /* 采用順序存儲方式存儲主串 s和模式 t, */ /* 若模式 t在主串 s中從第 pos位置開始有匹配的子串, */ /* 返回位置,否則返回 -1 */ char *p , *q ; int k , j ; k=pos-1 ; j=0 ; p=s.str+pos-1 ; q=t.str ; /* 初始匹配位置設(shè)置 */ /* 順序存放時第 pos位置的下標值為 pos-1 */ while (ks.length) q+ ; k+ ; j+ ; else k=k-j+1 ; j=0 ; q=t.str ; p=s.str+k ; /* 重新設(shè)置匹
53、配位置 */ if (j=t.length) return(k-t.length) ; / * 匹配,返回位置 */ else return(-1) ; /* 不匹配,返回 -1 */ 該算法簡單,易于理解。在一些場合的應(yīng)用里,如 文字處理中的文本編輯,其效率較高。 該算法的時間復(fù)雜度為 O(n*m) ,其中 n 、 m分別 是主串和模式串的長度。通常情況下,實際運行過程中, 該算法的執(zhí)行時間近似于 O(n+m) 。 理解該算法的關(guān)鍵點 當?shù)谝淮?sktj時:主串要退回到 k-j+1的位置,而模 式串也要退回到第一個字符(即 j=0的位置)。 比較出現(xiàn) sktj時:則應(yīng)該有 sk-1=tj-1
54、, , sk-j+1=t1, sk-j=t0 。 4.3.2 模式匹配的一種改進算法 該改進算法是由 D.E.Knuth , J.H.Morris和 V.R.Pratt提出來的,簡稱為 KMP算法。其 改進在于: 每當一趟匹配過程出現(xiàn)字符不相等時,主串指示器 不用回溯,而是利用已經(jīng)得到的“部分匹配”結(jié)果,將 模式串的指示器向右“ 滑動 ”盡可能遠的一段距離后, 繼續(xù)進行比較。 例: 設(shè)有串 s=abacabab , t=abab 。則第一次匹配 過程如圖 4-2所示。 圖 4-2 模式匹配示例 s=“a b cbb” i=3 | | 匹配失敗 t=“a b b” j=3 在 i=3和 j=3時
55、,匹配失敗。但重新開始第二次匹配 時,不必從 i=1 , j=0開始。因為 s1=t1, t0t1,必有 s1t0, 又因為 t0=t2, s2=t2,所以必有 s2=t0。由此可知,第二次 匹配可以直接從 i=3 、 j=1開始。 總之,在主串 s與模式串 t的匹配過程中,一旦出現(xiàn) sitj ,主串 s的指針不必回溯,而是直接與模式串的 tk(0 kj進行比較,而 k的取值與主串 s無關(guān),只與模式 串 t本身的構(gòu)成有關(guān),即從模式串 t可求得 k值。 ) 不失一般性,設(shè) 主串 s=s1s2 sn , 模式串 t=t1 t2 t m 。 當 sitj(1 i n-m, 1 jm, mn)時,主串
56、 s的指針 i 不必回溯,而模式串 t的指針 j回溯到第 k(kk滿足 4-1式。 t1t2t k-1= si-(k-1) si-(k-2) s i-2 si-1 (4-1) 而已經(jīng)得到的 “部分匹配”的結(jié)果為: tj-(k-1) tj-k t j-1=si-(k-1) si-(k-2) s i-2 si-1 (4-2) 由式 (4-1)和式 (4-2)得: t1t2t k-1=tj-(k-1) tj-k t j-1 (4-3) 該推導過程可用圖 4-3形象描述。實際上,式 (4-3) 描述了模式串中存在相互重疊的子串的情況。 圖 4-3 KMP算法示例 主串 s i 模式串 t k j 0
57、當 j=1時 Maxk|1kj t1t2t k-1=tj-(k-1) tj-k t j-1 該集合不空時 1 其它情況 nextj= 定義 nextj函數(shù)為 在求得了 nextj值之后, KMP算法的思想是: 設(shè)目標串 (主串 )為 s,模式串為 t ,并設(shè) i指針和 j指針 分別指示目標串和模式串中正待比較的字符,設(shè) i和 j的 初值均為 1。若有 si=tj,則 i和 j分別加 1。否則, i不變, j 退回到 j=nextj的位置,再比較 si和 tj,若相等,則 i和 j 分別加 1。否則, i不變, j再次退回到 j=nextj的位置, 依此類推。直到下列兩種可能: (1) j退回到
58、某個下一個 j值時字符比較相等,則指 針各自加 1繼續(xù)進行匹配。 (2)退回到 j=0,將 i和 j分別加 1,即從主串的下一個字 符 si+1模式串的 t1重新開始匹配。 KMP算法如下 #define Max_Strlen 1024 int nextMax_Strlen; int KMP_index (StringType s , StringType t) /* 用 KMP算法進行模式匹配,匹配返回位置,否則返回 -1 */ /*用靜態(tài)存儲方式保存字符串, s和 t分別表示主串和模式串 */ int k=0 , j=0 ; /*初始匹配位置設(shè)置 */ while (ks.length)
59、else return(-1) ; 很顯然, KMP_index函數(shù)是在已知下一個函數(shù)值的 基礎(chǔ)上執(zhí)行的,以下討論如何求 next函數(shù)值 ? 由式 (4-3)知,求模式串的 nextj值與主串 s無關(guān),只 與模式串 t本身的構(gòu)成有關(guān),則可把求 next函數(shù)值的問題 看成是一個模式匹配問題。由 next函數(shù)定義可知: 當 j=1時: next1=0。 設(shè) nextj=k,即在模式串中存在: t1t2t k-1=tj-(k-1)tj- k t j-1 ,其中下標 k滿足 1kk 滿足上式,即: nextj+1=nextj+1=k+1 (2) 若有 tktj :則表明在模式串中有: t1 t2t k
60、-1 tktj- (k-1) tj-k tj-1 tj ,當 tktj時應(yīng) 將模式向右滑動 至以模 式中的第 nextk個字符和主串中的第 j個字符相比較。 若 nextk= k,且 tj = tk,則說明在主串中第 j+1字符 之前存在一個長度為 k(即 nextk)的最長子串,與模 式串中從第一個字符起長度為 k的子串相等。即 nextj+1=k+1 同理,若 tjtk,應(yīng) 將模式繼續(xù)向右滑動 至將模式中 的第 nextk個字符和 tj對齊, ,依此類推,直到 tj和 模式串中的某個字符匹配成功或者不存在任何 k(1 kj)滿足等式: t1 t2t k-1 tk=tj-(k-1) tj-k
61、 tj-1 tj 則: nextj+1=1 根據(jù)上述分析, 求 next函數(shù)值的算法如下: void next(StringType t , int next) /* 求模式 t的 next串 t函數(shù)值并保存在 next數(shù)組中 */ int k=1 , j=0 ; next1=0; while (k1)個具有相同數(shù)據(jù)類型的數(shù)據(jù)元素 a1, a2, , an組成的有序序列 ,且該 序列必須存儲在一塊 地址連續(xù)的存儲單元中 。 數(shù)組中的數(shù)據(jù)元素 具有相同數(shù)據(jù)類型 。 數(shù)組是一種隨機存取結(jié)構(gòu),給定一組下標,就可 以訪問與其對應(yīng)的數(shù)據(jù)元素。 數(shù)組中的數(shù)據(jù)元素個數(shù)是固定的。 5.1.1 數(shù)組的 抽象數(shù)據(jù)
62、類型定義 1 抽象數(shù)據(jù)類型定義 ADT Array 數(shù)據(jù)對象: ji= 0,1, ,bi-1 , 1,2, ,n ; D = aj1j2 jn | n0稱為數(shù)組的維數(shù) , bi是數(shù)組第 i維的 長度 , ji是數(shù)組元素 第 i維的下標 , aj1j2 jn ElemSet 數(shù)據(jù)關(guān)系: R = R1, R2, , Rn Ri=|0 jk bk-1 , 1 k n且 ki, 0 ji bi-2, aj1j2 ji+1 jn D 基本操作: ADT Array 由上述定義知 , n維數(shù)組中有 b1b2 bn個數(shù)據(jù) 元素 , 每個數(shù)據(jù)元素都受到 n維關(guān)系的約束 。 2 直觀的 n維數(shù)組 以二維數(shù)組為例
63、討論。將 二維數(shù)組看成是一個定長 的線性表 , 其 每個元素又是一個定長的線性表 。 設(shè)二維數(shù)組 A=(aij)mn,則 A=(1, 2, , p) (p=m或 n) 其中每個數(shù)據(jù)元素 j是一個列向量 (線性表 ) : j =(a1j , a2j , , amj) 1 j n 或 是一個行向量 : i =(ai1 , ai2 , , ain) 1 i m 如圖 5-1所示。 a11 a12 a1n a21 a22 a2n am1 am2 amn A= A= a11 a12 a1n a21 a22 a2n am1 am2 amn a11 a21 am1 a12 a22 am2 a1n a2n a
64、mn A= 圖 5-1 二維數(shù)組圖 例形式 (a) 矩陣 表示形式 (b) 列向量的一維數(shù)組形式 (c) 行向量的一維數(shù)組形式 5.2 數(shù)組的 順序表示和實現(xiàn) 數(shù)組一般不做插入和刪除操作,也就是說,數(shù)組 一旦建立,結(jié)構(gòu)中的元素個數(shù)和元素間的關(guān)系就不再發(fā) 生變化。因此,一般都是 采用順序存儲的方法來表示數(shù) 組 。 問題 : 計算機的 內(nèi)存結(jié)構(gòu)是一維 (線性 )地址結(jié)構(gòu) , 對于多維數(shù)組,將其存放 (映射 )到內(nèi)存一維結(jié)構(gòu)時,有 個 次序約定問題 。即必須按某種次序?qū)?shù)組元素排成一 列序列,然后將這個線性序列存放到內(nèi)存中。 二維數(shù)組是最簡單的多維數(shù)組,以此為例說明多維 數(shù)組存放 (映射 )到內(nèi)存一
65、維結(jié)構(gòu)時的 次序約定問題 。 通常有兩種順序存儲方式 行優(yōu)先順序 (Row Major Order) : 將數(shù)組元素 按行排列,第 i+1個行向量緊接在第 i個行向量后面。對 二維數(shù)組,按行優(yōu)先順序存儲的線性序列為: a11,a12,a 1n, a21,a22,a 2n , a m1,am2,a mn PASCAL、 C是按行優(yōu)先順序存儲的,如圖 5-2(b) 示。 列優(yōu)先順序 (Column Major Order) : 將數(shù)組 元素按列向量排列,第 j+1個列向量緊接在第 j個列向量 之后,對二維數(shù)組,按列優(yōu)先順序存儲的線性序列為: a11,a21,a m1, a12,a22,a m2,
66、, a n1,an2,a nm FORTRAN是按列優(yōu)先順序存儲的,如圖 5-2(c)。 圖 5-2 二維數(shù)組及其順序存儲圖 例形式 a11 a12 a1n a21 a22 a2n am1 am2 amn A= (a) 二維數(shù)組的表示形式 (b) 行優(yōu)先順序存儲 (c) 列 優(yōu)先順序存儲 a11 a12 a1n 第 1 行 a21 a22 a2n 第 2 行 am1 am2 Amn 第 m 行 a11 a21 am1 第 1 列 a12 a22 am2 第 2 列 a1m a2m amn 第 n 列 設(shè)有二維數(shù)組 A=(aij)mn,若每個元素占用的存儲單 元數(shù)為 l(個 ), LOCa11表示元素 a11的首地址 ,即 數(shù)組 的 首地址 。 1 以 “ 行優(yōu)先順序 ” 存儲 第 1行中的 每個元素對應(yīng)的 (首 )地址是: LOCa1j=LOCa11+(j-1)l j=1,2, ,n (2) 第 2行中的 每個元素對應(yīng)的 (首 )地址是: LOCa2j=LOCa11+nl +(j-1)l j=1,2, ,n 第 m行中的 每個元素對應(yīng)的 (首 )地址是: LOCamj=LOCa11+(
- 溫馨提示:
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)容負責。
6. 下載文件中如有侵權(quán)或不適當內(nèi)容,請與我們聯(lián)系,我們立即糾正。
7. 本站不保證下載資源的準確性、安全性和完整性, 同時也不承擔用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。