鏈表基本操做實現

前言算法

這應該算是Mark Allen Weiss《數據結構和算法分析——C語言描述》的家庭做業。從前的學習筆記保存在OneNote,正好藉着寫博的機會翻出來複習。數據結構

正文數據結構和算法

鏈表的定義很簡單,每一個節點包含數據域和指針域。學習

下面是單鏈表的實現,包含list.h和list.c兩個文件。list.h是接口聲明,涉及的操做也會在這裏說明;list.c就是接口實現了。spa

  1 /*
  2   規範化單鏈表:
  3   
  4  (1)構建帶頭結點的鏈表
  5  (2)規範化操做集
  6 */
  7 
  8 /*接口聲明*/
  9 
 10 #include <stdbool.h>
 11 
 12 typedef int ElementType;
 13 
 14 #ifndef _LIST_H
 15 #define _LIST_H
 16 
 17 struct Node;
 18 
 19 typedef struct Node * PtrToNode;
 20 typedef PtrToNode List;
 21 typedef PtrToNode Position;
 22 
 23 /*操做集*/
 24 
 25 /*建立一個頭結點,並將其設爲空表*/
 26 List MakeEmpty(List L);
 27 
 28 /*判斷鏈表是否爲空表,若爲空表返回ture,不然返回false*/
 29 bool IsEmpty(List L);
 30 
 31 /*判斷位置P是否爲尾元素,如果返回ture,不然返回false*/
 32 bool IsLast(Position P,List L);
 33 
 34 /*查找元素在鏈表中的位置,若未找到返回NULL*/
 35 Position Find(ElementType X,List L);
 36 
 37 /*刪除元素,從鏈表中*/
 38 void Delete(ElementType X,List L);
 39 
 40 /*查找元素在鏈表中的位置,但返回上個元素的位置。若返回尾部說明未找到目標元素*/
 41 Position FindPrevious(ElementType X,List L);
 42 
 43 /*將元素插入指定位置P的後面*/
 44 void Isert(ElementType X,List L,Position P);
 45 
 46 /*刪除鏈表,但依然保留頭結點*/
 47 void DeleteList(List L);
 48 
 49 /*報告鏈表頭結點位置*/
 50 Position Header(List L);
 51 
 52 /*報告鏈表首元素位置*/
 53 Position First(List L);
 54 
 55 /*報告位置P的下個位置*/
 56 Position Advance(Position P);
 57 
 58 /*取出位置P中元素*/
 59 ElementType Retrieve(Position P);
 60 
 61 /*打印鏈表*/
 62 void PrintfList(List L);
 63 
 64 /*交換兩個相鄰元素,BeforeP指向相鄰元素前面的一個元素*/
 65 void SwapWithNext(Position BeforeP,List L); 
 66 
 67 /*打印鏈表L中由鏈表P所指定位置上的元素(L和P是升序鏈表)*/
 68 void PrintLots(List L,List P);
 69 
 70 /*求兩個升序鏈表的交集*/
 71 List Intersect(List L1,List L2);
 72 
 73 /*求兩個升序鏈表的並集*/
 74 List Union(List L1,List L2);
 75 
 76 /*將鏈表反轉,O(N)*/
 77 void Reverse(List L); 
 78 
 79 #endif 
 80 
 81 #include <stdio.h>
 82 #include <stdlib.h>
 83 #include "list.h"
 84 
 85 /*聲明特定的結構*/
 86 struct Node
 87 {
 88     ElementType Element;
 89     struct Node *  Next;
 90 };
 91 
 92 /*==================================================================================*/
 93 
 94 /*接口實現*/
 95 /*void MakeEmpty(List * L)
 96 {
 97    *L = (List)malloc(sizeof(struct Node)); 
 98     if (*L == NULL)
 99     {
100        printf("初始化失敗,退出程序\n");
101        exit(1);
102     }
103     
104     (*L)->Next = NULL;
105 }*/
106 
107  List MakeEmpty(List L)
108 {
109     if (L != NULL)
110         DeleteList(L);//若是鏈表非空,則刪除鏈表
111     L = (List)malloc(sizeof(struct Node));
112     if (L == NULL)
113     {
114        printf("初始化失敗,退出程序\n");
115        exit(1);
116     }
117       
118     L->Next = NULL;
119     return L;
120 }
121 
122 bool IsEmpty(List L)
123 {
124     return L->Next == NULL;
125 }
126 
127 bool IsLast(Position P,List L)
128 {
129     return P->Next == NULL;
130 }
131 
132 Position Find(ElementType X,List L)
133 {
134     Position P = L->Next;
135     
136     while (P != NULL && P->Element != X)
137         P = P->Next;
138         
139     return P;
140 }
141 
142 void Delete(ElementType X,List L)
143 {
144     Position LastElement = FindPrevious(X,L);
145     Position TheElement; 
146     
147     if (!IsLast(LastElement,L))    //LastElement不是尾部,說明找到X 
148     {
149        TheElement = LastElement->Next;
150        LastElement->Next = TheElement->Next;
151        free(TheElement);
152     }
153 }
154 
155 Position FindPrevious(ElementType X,List L)
156 {
157     Position P = L;
158     
159     while (P->Next != NULL && P->Next->Element != X)
160         P = P->Next;
161     
162     return P;
163 }
164 
165 void Isert(ElementType X,List L,Position P)
166 {
167     Position Temp;
168     
169     Temp = (List)malloc(sizeof(struct Node));
170     if (Temp == NULL)
171     {
172        printf("內存分配失敗,退出程序\n");
173        exit(1);
174     }
175     
176     Temp->Element = X;
177     
178     Temp->Next = P->Next;
179     P->Next = Temp;
180 }
181 
182 void DeleteList(List L)
183 {
184     Position P = L->Next;
185     Position Temp;
186     
187     L->Next = NULL;       //保留頭結點 
188     
189     while (P != NULL)
190     {
191         Temp = P->Next;
192         free(P);
193         P = Temp;
194     }
195 }
196 
197 Position Header(List L)
198 {
199     return L;
200 }
201 
202 Position First(List L)
203 {
204     return L->Next;
205 }
206 
207 Position Advance(Position P)
208 {
209     return P->Next;
210 }
211 
212 ElementType Retrieve(Position P)
213 {
214     return P->Element;
215 }
216 
217 void PrintfList(List L)
218 {
219     Position P;
220     
221     if (IsEmpty(L))
222        printf("List is empty!\n");
223     
224     P = First(L);
225     while (P != NULL)
226     {
227         printf("%3d",Retrieve(P));
228         P = Advance(P);
229     }
230     printf("\n");
231 }
232 
233 void SwapWithNext(Position BeforeP,List L)
234 {
235     Position P;
236     Position AfterP;
237     
238     P = Advance(BeforeP);
239     if (P != NULL)
240     {
241        AfterP = Advance(P);
242        if (AfterP != NULL)
243        {
244            P->Next = AfterP->Next;
245            BeforeP->Next = AfterP;
246            AfterP->Next = P;
247        }
248     }
249 }
250 
251 void PrintLots(List L,List P)
252 {
253     int count = 1;//鏈表L元素計數器,初始值指向第一個元素
254     Position LTemp;
255     Position PTemp;
256     
257     LTemp = First(L);
258     PTemp = First(P);
259     
260     /*當鏈表L和P任一爲空中止打印*/
261     while (LTemp != NULL && PTemp != NULL)
262     {
263         if (count == PTemp->Element)   //判斷是否打印當前元素
264         {
265            printf("%3d",LTemp->Element);
266            PTemp = Advance(PTemp);     //指向下個要打印的目標元素 
267         } 
268         
269         LTemp = Advance(LTemp);
270         count++;
271     }
272     printf("\n"); 
273 }
274 
275 
276 List Intersect(List L1,List L2)
277 {
278     List Result;
279     Position Ptr1;
280     Position Ptr2;
281     Position P;
282     
283     Result = MakeEmpty(NULL);
284     Ptr1 = First(L1);
285     Ptr2 = First(L2);
286     P = Header(Result);
287     
288     /*歸併操做*/
289     while (Ptr1 != NULL && Ptr2 != NULL)
290     {
291         if (Ptr1->Element < Ptr2->Element)
292            Ptr1 = Advance(Ptr1);
293         
294         else if (Ptr1->Element > Ptr2->Element)
295            Ptr2 = Advance(Ptr2);
296            
297         else
298         {
299            Isert(Ptr1->Element,Result,P);
300            
301            Ptr1 = Advance(Ptr1);
302            Ptr2 = Advance(Ptr2);
303            P = Advance(P);
304         }
305     }
306     
307     return Result; 
308 }
309 
310 
311 
312 List Union(List L1,List L2)
313 {
314     List Result;
315     Position Ptr1;
316     Position Ptr2;
317     Position P;
318     
319     Result = MakeEmpty(NULL);
320     Ptr1 = First(L1);
321     Ptr2 = First(L2);
322     P = Header(Result);
323     
324     /*歸併操做*/
325     while (Ptr1 != NULL&& Ptr2 != NULL)
326     {
327         if (Ptr1->Element < Ptr2->Element)
328         {
329            Isert(Ptr1->Element,Result,P);
330            Ptr1 = Advance(Ptr1);
331         }
332         
333         else if (Ptr1->Element > Ptr2->Element)
334         {
335            Isert(Ptr2->Element,Result,P);
336            Ptr2 = Advance(Ptr2);
337         }
338         
339         else
340         {
341            Isert(Ptr1->Element,Result,P);
342            Ptr1 = Advance(Ptr1);
343            Ptr2 = Advance(Ptr2);
344         }
345         
346         P = Advance(P);
347     }
348     
349     while (Ptr1 != NULL)
350     {
351         Isert(Ptr1->Element,Result,P);
352         Ptr1 = Advance(Ptr1);
353         P = Advance(P);
354     }
355     
356     while (Ptr2 != NULL)
357     {
358         Isert(Ptr2->Element,Result,P);
359         Ptr2 = Advance(Ptr2);
360         P = Advance(P);
361     }
362     
363     return Result;
364 }
365 
366 void Reverse(List L)
367 {
368     Position Temp;
369     Position P;
370     
371     if (IsEmpty(L) || L->Next->Next == NULL)
372        return;
373     
374     P = First(L);
375     L->Next = NULL;
376     
377     while (P != NULL)
378     {
379         Temp = Advance(P);
380         P->Next = First(L);
381         L->Next = P;
382         
383         P = Temp;
384     }
385 }
相關文章
相關標籤/搜索