深入解析双向链表与单向链表的区别:示例详解
创作时间:
作者:
@小白创作中心
深入解析双向链表与单向链表的区别:示例详解
引用
CSDN
1.
https://blog.csdn.net/qq_35320456/article/details/140401203
链表是一种灵活的数据结构,它通过指针连接一系列节点,实现了动态数组的特性。在众多链表类型中,单向链表和双向链表是最为常见的两种。本文将重点探讨这两种链表的区别,并通过示例进行详细说明。
一、单向链表与双向链表的定义及结构
单向链表
定义:单向链表是链表的一种,每个节点包含数据域和一个指向下一个节点的指针。
结构示例:
struct Node {
int data; // 数据域
Node next; // 指针域,指向下一个节点
}
双向链表
定义:双向链表是链表的一种,每个节点包含数据域、一个指向前一个节点的指针和一个指向下一个节点的指针。
结构示例:
struct Node {
int data; // 数据域
Node prev; // 指针域,指向前一个节点
Node next; // 指针域,指向下一个节点
}
二、单向链表与双向链表的区别示例
插入操作
假设要在链表中的某个节点后插入一个新的节点,以下是单向链表和双向链表的插入操作示例。
单向链表插入操作:
// 假设要在节点p后插入新节点newNode
newNode.next = p.next;
p.next = newNode;
双向链表插入操作:
// 假设要在节点p后插入新节点newNode
newNode.prev = p;
newNode.next = p.next;
if (p.next != null) {
p.next.prev = newNode;
}
p.next = newNode;
从上述示例可以看出,双向链表在插入操作时需要同时维护前驱节点和后继节点的指针,而单向链表只需维护后继节点的指针。
删除操作
假设要删除链表中的某个节点,以下是单向链表和双向链表的删除操作示例。
单向链表删除操作:
// 假设要删除节点p
if (p.prev != null) {
p.prev.next = p.next;
}
双向链表删除操作:
// 假设要删除节点p
if (p.prev != null) {
p.prev.next = p.next;
} else {
// p是头节点,需要更新头节点
head = p.next;
}
if (p.next != null) {
p.next.prev = p.prev;
}
从上述示例可以看出,双向链表在删除操作时需要同时维护前驱节点和后继节点的指针,而单向链表只需维护前驱节点的指针。
三、完整示例
以下是双向链表的一个简单示例(C++):
#include <iostream>
struct Node {
int data;
Node* next;
Node* prev;
Node(int data) : data(data), next(nullptr), prev(nullptr) {}
};
class DoublyLinkedList {
public:
Node* head;
Node* tail;
DoublyLinkedList() : head(nullptr), tail(nullptr) {}
void insertAtHead(int data) {
Node* newNode = new Node(data);
if (head == nullptr) {
head = tail = newNode;
} else {
newNode->next = head;
head->prev = newNode;
head = newNode;
}
}
void printList() const {
Node* temp = head;
while (temp != nullptr) {
std::cout << temp->data << " <-> ";
temp = temp->next;
}
std::cout << "NULL" << std::endl;
}
~DoublyLinkedList() {
Node* current = head;
while (current != nullptr) {
Node* next = current->next;
delete current;
current = next;
}
}
};
int main() {
DoublyLinkedList list;
list.insertAtHead(3);
list.insertAtHead(2);
list.insertAtHead(1);
list.printList();
return 0;
}
在这个示例中,我们创建了一个双向链表,并在链表头部插入新结点。注意,插入操作中需要同时更新新结点的next和prev指针,以及前一个结点的prev指针和后一个结点的next指针。
以下是单向链表的一个简单示例(C++):
#include <iostream>
struct Node {
int data;
Node* next;
Node(int data) : data(data), next(nullptr) {}
};
class LinkedList {
public:
Node* head;
LinkedList() : head(nullptr) {}
void insertAtHead(int data) {
Node* newNode = new Node(data);
newNode->next = head;
head = newNode;
}
void printList() const {
Node* temp = head;
while (temp != nullptr) {
std::cout << temp->data << " -> ";
temp = temp->next;
}
std::cout << "NULL" << std::endl;
}
~LinkedList() {
Node* current = head;
while (current != nullptr) {
Node* next = current->next;
delete current;
current = next;
}
}
};
int main() {
LinkedList list;
list.insertAtHead(3);
list.insertAtHead(2);
list.insertAtHead(1);
list.printList();
return 0;
}
四、总结
通过以上示例,我们可以总结出单向链表与双向链表以下几个主要区别:
- 结构差异:双向链表每个节点有两个指针域,分别指向前一个节点和后一个节点;单向链表只有一个指针域,指向下一个节点。
- 操作差异:双向链表在插入和删除操作时,需要同时维护前驱节点和后继节点的指针;单向链表只需维护一个方向的指针。
- 功能差异:双向链表可以快速访问前驱节点和后继节点,适用于需要双向遍历的场景;单向链表只能单向遍历,适用于只需单向访问的场景。
- 空间占用:双向链表由于每个节点有两个指针域,因此占用空间较大;单向链表占用空间较小。
- 实现复杂度:双向链表实现相对复杂,需要考虑前驱节点和后继节点的指针修改;单向链表实现简单。
在实际应用中,根据具体需求和场景选择合适的链表类型,可以提高程序的性能和效率。
热门推荐
企业该如何做好固定资产管理?
C语言如何求极值
文化湿地:江心岛书院有个“岛师团”
格力集团是什么级别?与格力电器什么关系?
皮皮虾的美味烹饪秘籍
张一鸣成首富,这个时代太疯狂?
自行车改装完全指南:从入门到精通的工具清单
护士长陈君燕:六次捐献“熊猫血”,为生命续航
颠覆传统:重新探索《小红帽》--不平凡的故事中藏着的善良与勇气
独立思考的第一步:如何区分“观点“与“事实”?
“og”一词:从嘻哈文化到网络热梗的演变之路
2025年蛇年速成宝典:对仗句怎么写?从入门到惊艳!
西汉的兴衰转折
一使劲小肚子疼?自查步骤与就医指南
研究发现ChatGPT在情感反应的测试中更富有同情心
离婚率高居不下:经济独立真的是罪魁祸首吗?
什么叫股票持仓?股票持仓的风险如何控制?
工程公差:类型、基本规则和配合
哪吒到底从何而来?一文追溯哪吒的前世今生!
上传速度如何测试软件
营养学视角下的粥食:从食材到搭配的全面解析
储藏间设计装修全攻略:从空间规划到细节处理
项目管理会议纪要怎么写
西医开中成药须通过培训 中国用药日趋合理规范
英国G5超级精英大学指的什么?
用科技照亮敦煌的未来千年
硒元素每日推荐摄入60到400微克,如何补充,通过吃!
头皮有红点是怎么回事
如何预测游戏比分数据库
老人一餐只需15元,16种菜可选择,这家老年餐厅羡煞旁人