深入解析双向链表与单向链表的区别:示例详解
创作时间:
作者:
@小白创作中心
深入解析双向链表与单向链表的区别:示例详解
引用
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;
}
四、总结
通过以上示例,我们可以总结出单向链表与双向链表以下几个主要区别:
- 结构差异:双向链表每个节点有两个指针域,分别指向前一个节点和后一个节点;单向链表只有一个指针域,指向下一个节点。
- 操作差异:双向链表在插入和删除操作时,需要同时维护前驱节点和后继节点的指针;单向链表只需维护一个方向的指针。
- 功能差异:双向链表可以快速访问前驱节点和后继节点,适用于需要双向遍历的场景;单向链表只能单向遍历,适用于只需单向访问的场景。
- 空间占用:双向链表由于每个节点有两个指针域,因此占用空间较大;单向链表占用空间较小。
- 实现复杂度:双向链表实现相对复杂,需要考虑前驱节点和后继节点的指针修改;单向链表实现简单。
在实际应用中,根据具体需求和场景选择合适的链表类型,可以提高程序的性能和效率。
热门推荐
港币如何安全转换为美元?这种安全转换有哪些注意事项?
讣告写作指南:格式、语言与注意事项全解析
夫妻共同理财怎么处理?如何实现理财目标并防范风险?
小型面条机家用选购新视角
绿美文山:生态攻坚战正酣
黄金作为硬通货的原因、特点及其经济影响
恶意别车怎么处罚?一文详解相关法规与应对措施
人到中年,最抗衰老的 4 大运动
防冻液不足该如何补充?怎样判断防冻液是否需要更换?
新风系统与排风扇有什么不同?
宇宙文明的七个等级:从行星文明到全能文明
體味好重好尷尬?該怎麼吃才能幫助減輕體味問題?
老挝结婚手续:了解相关流程与要求
运动有助于改善情绪吗
拔罐后的罐印都暗示啥?什么人不能拔火罐?来看→
俄罗斯到底是个什么样的国家?俄罗斯的人民对我们又是什么态度?
肾衰竭引起的脚浮肿:原因、治疗与日常管理
开关电源工作原理详解:从基本原理到各类电路类型
惯性的大小和定义
研二还是研三找工作?提前规划,抢占“就业黄金期”关键节点!
经常嘴角周围出现小水泡?疼痛难忍?是单纯疱疹在作怪
牛奶是养胃的还是伤胃的?
【健康科普】最新高血压防治指南发布——血压达到这个数,说明你血压高了!
塔山阻击战,为何被国民党称为“最不可思议的失败”
灌香肠,吃“头菜”,“打囤子”……南通的“年味”有专属仪式感
武汉新汉阳站迎来重大进展,将实现与多条高铁线路互联互通
如何制定高效的社交媒体备份策略:5个必备步骤
沉香树的生长环境与特点(了解沉香树的适生环境)
如何清除Windows 10指纹数据库
大头娃娃事件全解析:现状如何?