158
技術社區[雲棲]
雙向鏈表的C實現
雙向鏈表需要定義一個結構體,結構體有3個屬性
typedef struct __Node{
int data; 數據
struct __Node *pre; 指向前一個結點指針
struct __Node *next; 指向下一個結點指針
}Node;
其中 pre和next指針是嵌套定義。
一般鏈表定義一個頭指針
Node *head;
指向鏈表第一個結點,如果鏈表為空的話,那麼head == NULL。
雙向鏈表一般分為init,insert, delete, search, destroy等幾種操作
1、init
初始化:將頭指針head置為NULL即可
2、insert
插入:這裏我隻實現了在表頭位置插入新元素。在表頭位置插入元素的話,需要注意區別處理空表和非空表的情況。
1)空表的話,因為有init過程,所以head為NULL,新元素的next指針指向head(NULL),同時設置新元素的pre指針為NULL即可。
2)非空表的話,需要設置以前的頭結點元素的pre指針,因為頭結點的pre指針都是NULL。其他和空表一樣操作。
如果要實現在鏈表的指定位置插入的話,需要先遍曆鏈表找到那個位置,然後再插入。
3、delete
刪除:這裏我實現了刪除第一個元素和刪除指定的元素。其中刪除指定的元素需要查找。
4、search
查找:遍曆鏈表進行查找即可
5、destroy
銷毀:將每個結點開辟的內存分別釋放,最後將頭結點指針head置為NULL即可。
在以上操作過程中,要注意當鏈表非空時,一定要保證尾結點的next域為NULL,頭結點的pre域為NULL,否則容易導致錯誤!
C語言實現代碼如下:
- #include <stdio.h>
- #include <stdlib.h>
- //定義結點
- typedef struct __Node{
- int data;
- struct __Node *pre;
- struct __Node *next;
- }Node;
- //定義帶頭結點的雙向鏈表
- typedef struct __doublyLinkedList{
- Node * head;
- }dLinkedList;
- //初始化:頭結點置空
- void init(dLinkedList *L){
- if(L == NULL){
- printf("鏈表異常/n");
- return;
- }
- L->head = NULL;
- }
- //
- void insert(dLinkedList *L, int data){
- Node *p = NULL;
- if(L == NULL){
- printf("雙向鏈表不存在/n");
- return;
- }
- p = (Node*)malloc(sizeof(Node));
- if(p == NULL){
- printf("內存分配失敗!/n");
- return;
- }
- p->data = data;
- p->next = L->head;
- if(L->head != NULL){
- L->head->pre = p;
- }
- p->pre = NULL;
- L->head = p;
- }
- Node *search(dLinkedList L, int data){
- Node *p = NULL;
- if(L.head == NULL){
- printf("鏈表為空/n");
- return NULL;
- }
- p = L.head;
- while(p != NULL && p->data != data){
- p = p->next;
- }
- if(p != NULL){
- printf("查找值為 %d的元素成功/n", data);
- return p;
- }
- else{
- printf("查找值為 %d的元素失敗/n", data);
- return NULL;
- }
- }
- void deleteFirstData(dLinkedList *L){
- Node *p = NULL;
- if(L == NULL){
- printf("雙向鏈表異常!/n");
- return;
- }
- if(L->head == NULL){
- printf("雙向鏈表為空!/n");
- return;
- }
- p = L->head;
- //only one element
- if(p->next == NULL){
- L->head = NULL;
- free(p);
- p = NULL;
- printf("成功刪除第一個元素!/n");
- return;
- }
- else{
- L->head = p->next;
- L->head->pre = NULL;
- free(p);
- p = NULL;
- printf("成功刪除第一個元素!/n");
- return;
- }
- }
- void deleteData(dLinkedList *L, int data){
- Node *p = NULL;
- Node *pre = NULL;
- printf("刪除查找中.../n");
- if((p = search(*L, data)) == NULL){
- printf("刪除值為 %d的元素失敗/n", data);
- return;
- }
- else{
- //first element
- if(p == L->head){
- deleteFirstData(L, data);
- return;
- }
- //last element
- else if(p->next == NULL){
- pre = p->pre;
- pre->next = NULL;
- free(p);
- p = NULL;
- printf("刪除最後一個元素成功/n/n");
- return;
- }
- else{
- pre = p->pre;
- pre->next = p->next;
- p->next->pre = pre;
- free(p);
- p = NULL;
- printf("刪除值為 %d的元素成功/n/n", data);
- return;
- }
- }
- }
- void traversal(dLinkedList L){
- Node *p = NULL;
- if(L.head == NULL){
- printf("雙向鏈表為空!/n");
- return;
- }
- p = L.head;
- while(p != NULL){
- printf("%d ", p->data);
- p = p->next;
- }
- printf("/n遍曆成功/n/n");
- }
- void destroy(dLinkedList *L){
- Node *p = NULL;
- Node *temp = NULL;
- if(L == NULL){
- printf("鏈表異常/n");
- return;
- }
- printf("開始銷毀鏈表.../n");
- p = L->head;
- while(p != NULL){
- temp = p;
- p = p->next;
- free(temp);
- temp = NULL;
- }
- L->head = NULL;
- printf("銷毀成功!/n");
- }
- int main(int argc, char **args){
- dLinkedList L;
- int i;
- memset(&L, 0, sizeof(dLinkedList));
- init(&L);
- traversal(L);
- for(i=0;i<20;++i){
- insert(&L, i);
- }
- traversal(L);
- deleteFirstData(&L);
- deleteFirstData(&L);
- traversal(L);
- deleteFirstData(&L);
- deleteData(&L, 10);
- deleteData(&L, 16);
- deleteData(&L, 0);
- traversal(L);
- insert(&L, 100);
- insert(&L, 99);
- insert(&L, 98);
- insert(&L, 97);
- traversal(L);
- deleteData(&L, 99);
- deleteData(&L, 97);
- deleteData(&L, 96);
- traversal(L);
- destroy(&L);
- return 0;
- }
輸出如下
雙向鏈表為空!
19 18 17 16 15 14 13 12 11 10 9 8 7 6 5 4 3 2 1 0
遍曆成功
成功刪除第一個元素!
成功刪除第一個元素!
17 16 15 14 13 12 11 10 9 8 7 6 5 4 3 2 1 0
遍曆成功
成功刪除第一個元素!
刪除查找中...
查找值為 10的元素成功
刪除值為 10的元素成功
刪除查找中...
查找值為 16的元素成功
成功刪除第一個元素!
刪除查找中...
查找值為 0的元素成功
刪除最後一個元素成功
15 14 13 12 11 9 8 7 6 5 4 3 2 1
遍曆成功
97 98 99 100 15 14 13 12 11 9 8 7 6 5 4 3 2 1
遍曆成功
刪除查找中...
查找值為 99的元素成功
刪除值為 99的元素成功
刪除查找中...
查找值為 97的元素成功
成功刪除第一個元素!
刪除查找中...
查找值為 96的元素失敗
刪除值為 96的元素失敗
98 100 15 14 13 12 11 9 8 7 6 5 4 3 2 1
遍曆成功
開始銷毀鏈表...
銷毀成功!
Press any key to continue
最後更新:2017-04-03 05:39:11