双向链表(1) - 基本介绍以及插入节点
来源:互联网 发布:淘宝开店链接 编辑:程序博客网 时间:2024/05/01 05:15
双向链表(doubly linked list - DLL)的操作,与单链表很大程度上有相似之处。在开始本篇文章前,可以先回顾下单链表的类似操作。
参考单链表系列中的这两篇文章:”链表(1) - 介绍“, ”链表(3) - 插入节点“。
一个双向链表包含一个额外的指针, 称之为前向指针(prev pointer),与单链表中的后向指针(next pointer)一起来标识一个节点。
下面是使用C++代码来表示一个DLL的例子:
与单链表相比,双向链表有下面的这些优点和缺点。
2) DLL中的删除操作更有效,如果提供了要删除节点的指针。
这是因为,在单链表中如果要删除一个节点,则必须要知道前一个节点。有时为了得到这前一个节点,需要遍历整个链表,而在双向链表中,使用前向指针就可以很方便的得到前一个节点。
其实可以使用一个指针来实现双向链表。具体可以参考"高级数据结构"系列中的下面这两篇文章:
"高级链表 - 异或链表(1)" 以及 "高级链表 - 异或链表(2)"。
2) 所有的操作,都需要维护前向指针。例如 ,插入操作时,需要同时更改前向和后向指针。
1) 在DLL的头部
2) 在一个指定节点的后面
3) 在DLL的尾部
4) 在一个指定节点的前面
下面是具体操作的5个步骤.
根据step4, step5可知,修改后向指针时,是从右到左的顺序,即先改变新节点的后向指针,再改变给定节点的后向指针。
而根据step6, step7可知,修改前向指针时,是从左到右的顺序,即先改变新节点的前向指针,再改变后续节点的前向指针。
因为通常一个链表是用头节点来表示的,所以必须遍历整个链表,将最后一个节点的后向指针置为新节点。
下面是具体的7个实现步骤:
Created DLL is:
Traversal in forward direction
1 7 9 8 6 4
Traversal in reverse direction
4 6 8 9 7 1
参考单链表系列中的这两篇文章:”链表(1) - 介绍“, ”链表(3) - 插入节点“。
一个双向链表包含一个额外的指针, 称之为前向指针(prev pointer),与单链表中的后向指针(next pointer)一起来标识一个节点。
下面是使用C++代码来表示一个DLL的例子:
//双向链表中的节点元素struct Node{ int data; Node *next; // 指向下一个节点 Node *prev; // 指向前一个节点};
与单链表相比,双向链表有下面的这些优点和缺点。
相比单链表的一些优势
1) DLL支持正向和逆向的遍历方式。2) DLL中的删除操作更有效,如果提供了要删除节点的指针。
这是因为,在单链表中如果要删除一个节点,则必须要知道前一个节点。有时为了得到这前一个节点,需要遍历整个链表,而在双向链表中,使用前向指针就可以很方便的得到前一个节点。
相比单链表的一些弱势
1) DLL的每个节点,需要额外空间来保存前向指针。其实可以使用一个指针来实现双向链表。具体可以参考"高级数据结构"系列中的下面这两篇文章:
"高级链表 - 异或链表(1)" 以及 "高级链表 - 异或链表(2)"。
2) 所有的操作,都需要维护前向指针。例如 ,插入操作时,需要同时更改前向和后向指针。
DLL的插入操作
可以使用4种方式添加一个节点:1) 在DLL的头部
2) 在一个指定节点的后面
3) 在DLL的尾部
4) 在一个指定节点的前面
链表头部插入一个节点: (5个步骤)
新的节点通常添加到DLL的头部前面,并且变成新的头部节点。例如,对于一个双向链表10<->15<->20<->25,在头部插入一个节点5, 则会变成了5<->10<->15<->20<->25. 假设在链表头部进行节点插入的函数称之为push()。则这个push函数需要知道头指针,因为push必须将头指针指向新的节点。下面是具体操作的5个步骤.
// 给定链表的头指针(head)以及一个整数,插入一个新的节点至链表的头部// 之所以传入双指针,因为函数中需要修改链表void push(Node** head, int newData){//1. 分配新节点内存Node* newNode = new Node;//2. 赋值newNode->data = newData;//3. 将原始头节点做为新节点的后向指针,而前向指针置为NULLnewNode->next = (*head);newNode->prev = NULL;//4. 将原始头节点的前向指针置为新的节点if ((*head) != NULL)(*head)->prev = newNode;//5. 将头指针置为新的节点(*head) = newNode;}上面的前4个步骤,与单链表中插入节点至头部的操作步骤是一样的。只是这里新增了一个步骤,就是改变头部的前向指针。
指定节点的后面插入新节点: (7个步骤)
假设指定的节点为 prevNode, 然后在此节点的后面插入新的节点。//插入一个节点至指定节点的后面void insertAfter(Node* prevNode, int newData){// 1. 检查指定节点是否为NULLif (prevNode == NULL){std::cout<<"the given previous node cannot be NULL";return;}// 2. 分配新节点内存Node* newNode = new Node;// 3. 赋值newNode->data = newData;// 4. 将指定节点的后向指针,做为新节点的后向指针newNode->next = prevNode->next;// 5. 将新节点做为指定节点的后向指针prevNode->next = newNode;// 6. 将指定节点做为新节点的前向指针newNode->prev = prevNode;// 7. 调整新节点的后续节点的前向指针if (newNode->next != NULL)newNode->next->prev = newNode;}上面的前5个步骤,与单链表中插入节点至指定节点的后面的操作步骤是一样的。只是这里新增了两个步骤。即新节点的前向指针,以及新节点的后续节点的前向指针。
根据step4, step5可知,修改后向指针时,是从右到左的顺序,即先改变新节点的后向指针,再改变给定节点的后向指针。
而根据step6, step7可知,修改前向指针时,是从左到右的顺序,即先改变新节点的前向指针,再改变后续节点的前向指针。
链表的尾部插入新节点: (7个步骤)
这种情况下,新节点通常插入到最后一个节点的后面。 例如,对于双向链表5<->10<->15<->20<->25,在尾部插入新的节点30,则链表最终变成5<->10<->15<->20<->25<->30。因为通常一个链表是用头节点来表示的,所以必须遍历整个链表,将最后一个节点的后向指针置为新节点。
下面是具体的7个实现步骤:
// 给定链表的头指针(head)以及一个整数,插入一个新的节点至链表的尾部void append(Node** head, int newData){// 1. 分配新节点内存Node *newNode = new Node;Node *last = *head; //链表的尾部指针,用于step5// 2. 赋值newNode->data = newData;// 3. 新节点将成为尾节点,所以后向指针为NULLnewNode->next = NULL;// 4. 如果是空链表,则直接将新节点设置为头节点if (*head == NULL){newNode->prev = NULL;*head = newNode;return;}// 5. 如果不是空链表,则遍历链表,获取尾节点while (last->next != NULL)last = last->next;// 6. 修改尾节点的后向指针为新节点last->next = newNode;// 7. 修改新节点的前向指针为原始尾节点newNode->prev = last;return;}上面的前7个步骤,与单链表中进行相应操作的前6个步骤相同。新增的1个步骤是修改新节点的前向指针。
指定节点的前面插入新节点:(7个步骤)
假设指定的节点为 nextNode, 然后在此节点的前面插入新的节点。void insertBefore(Node* nextNode, int newData){// 1. 检查指定节点是否为NULLif (nextNode == NULL){printf("the given previous node cannot be NULL");return;}// 2. 分配新节点内存Node* newNode = new Node;// 3. 赋值newNode->data = newData;// 4. 将指定节点的前向指针,做为新节点的前向指针newNode->prev = nextNode->prev;// 5. 将新节点做为指定节点的前向指针nextNode->prev = newNode;// 6. 将指定节点做为新节点的后向指针newNode->next = nextNode;// 7. 调整新节点的前面节点的后向指针if (newNode->prev != NULL)newNode->prev->next = newNode;}
实现所有插入操作的完整程序
#include <iostream>struct Node{int data;Node *next; // 指向下一个节点Node *prev; // 指向前一个节点};// 给定链表的头指针(head)以及一个整数,插入一个新的节点至链表的头部// 之所以传入双指针,因为函数中需要修改链表void push(Node** head, int newData){//1. 分配新节点内存Node* newNode = new Node;//2. 赋值newNode->data = newData;//3. 将原始头节点做为新节点的后向指针,而前向指针置为NULLnewNode->next = (*head);newNode->prev = NULL;//4. 将原始头节点的前向指针置为新的节点if ((*head) != NULL)(*head)->prev = newNode;//5. 将头指针置为新的节点(*head) = newNode;}//插入一个节点至指定节点的后面void insertAfter(Node* prevNode, int newData){// 1. 检查指定节点是否为NULLif (prevNode == NULL){std::cout<<"the given previous node cannot be NULL";return;}// 2. 分配新节点内存Node* newNode = new Node;// 3. 赋值newNode->data = newData;// 4. 将指定节点的后向指针,做为新节点的后向指针newNode->next = prevNode->next;// 5. 将新节点做为指定节点的后向指针prevNode->next = newNode;// 6. 将指定节点做为新节点的前向指针newNode->prev = prevNode;// 7. 调整新节点的后续节点的前向指针if (newNode->next != NULL)newNode->next->prev = newNode;}// 给定链表的头指针(head)以及一个整数,插入一个新的节点至链表的尾部void append(Node** head, int newData){// 1. 分配新节点内存Node *newNode = new Node;Node *last = *head; //链表的尾部指针,用于step5// 2. 赋值newNode->data = newData;// 3. 新节点将成为尾节点,所以后向指针为NULLnewNode->next = NULL;// 4. 如果是空链表,则直接将新节点设置为头节点if (*head == NULL){newNode->prev = NULL;*head = newNode;return;}// 5. 如果不是空链表,则遍历链表,获取尾节点while (last->next != NULL)last = last->next;// 6. 修改尾节点的后向指针为新节点last->next = newNode;// 7. 修改新节点的前向指针为原始尾节点newNode->prev = last;return;}//插入一个节点至指定节点的前面void insertBefore(Node* nextNode, int newData){// 1. 检查指定节点是否为NULLif (nextNode == NULL){printf("the given previous node cannot be NULL");return;}// 2. 分配新节点内存Node* newNode = new Node;// 3. 赋值newNode->data = newData;// 4. 将指定节点的前向指针,做为新节点的前向指针newNode->prev = nextNode->prev;// 5. 将新节点做为指定节点的前向指针nextNode->prev = newNode;// 6. 将指定节点做为新节点的后向指针newNode->next = nextNode;// 7. 调整新节点的前面节点的后向指针if (newNode->prev != NULL)newNode->prev->next = newNode;}void printList(Node *head){Node *last = NULL;std::cout<<"\nTraversal in forward direction \n";while (head != NULL){std::cout<<" "<<head->data<<" ";last = head;head = head->next;}std::cout<<"\nTraversal in reverse direction \n";while (last != NULL){std::cout << " " << last->data << " ";last = last->prev;}std::cout << std::endl;}int main(){//初始化为空链表Node* head = NULL;// 插入节点6. 链表变为:6->NULLappend(&head, 6);// 插入节点7,链表变为:7->6->NULLpush(&head, 7);// 头部插入节点1,链表变为:1->7->6->NULLpush(&head, 1);// 尾部插入节点4,链表变为:1->7->6->4->NULLappend(&head, 4);// 在节点7后面插入节点8,链表变为:1->7->8->6->4->NULLinsertAfter(head->next, 8); // 节点8之前插入节点9,链表变为:1->7->9->8->6->4->NULLinsertBefore(head->next->next, 9);std::cout<<"Created DLL is: ";printList(head);return 0;}输出:
Created DLL is:
Traversal in forward direction
1 7 9 8 6 4
Traversal in reverse direction
4 6 8 9 7 1
0 0
- 双向链表(1) - 基本介绍以及插入节点
- 双向链表 删除节点 插入节点
- 双向链表的插入以及删除
- 双向链表基本操作:删除、插入、双向输出
- C语言实现双向链表删除节点、插入节点、双向输出等操作
- 关于双向链表插入节点的问题
- java建立双向链表,插入结点,删除节点
- 双向链表的建立、删除/插入节点
- 双向链表的插入与删除节点
- 实现双向链表删除一个节点P,在节点P后插入一个节点
- 不带头节点的双向循环链表基本操作
- 双向链表删除节点
- 链表指针新理解以及插入里链表节点新方法
- C语言实现双向非循环链表(不带头结点)的节点插入
- C语言实现双向非循环链表(带头结点尾结点)的节点插入
- 数据结构——10 双向链表插入和删除节点
- 双向链表的建立,添加节点和删除节点(注意插入和删除要分三种情况)
- 双向链表的初始化,建立,添加节点和删除节点(注意插入和删除要分三种情况)
- poj_1860
- 论ERP系统开发 二:框架结构
- 机器学习实战——条件随机场(CRF)
- 搞定linux上MySQL编程(六):C语言编写MySQL程序(结)
- oralce用存储过程实现分页 以及 用java调用这个存储过程的代码
- 双向链表(1) - 基本介绍以及插入节点
- hznu 1139: Minimax Triangulation(dp,三角形面积模板)
- 中文版android开发资源
- jquery 初步(四)内容过滤器
- hiho第五十周——欧拉路·二(Fleury算法求欧拉路径)
- Unity3d NGUI的使用(三)(UIButton及Anchor实际使用)
- java 基础学习-06 集合的使用 一
- 深度剖析malloc、free和new、delete
- hznu 1636: 方格取数(dp,灵活题)