深入解析双向链表与单向链表的区别:示例详解
创作时间:
作者:
@小白创作中心
深入解析双向链表与单向链表的区别:示例详解
引用
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;
}
四、总结
通过以上示例,我们可以总结出单向链表与双向链表以下几个主要区别:
- 结构差异:双向链表每个节点有两个指针域,分别指向前一个节点和后一个节点;单向链表只有一个指针域,指向下一个节点。
- 操作差异:双向链表在插入和删除操作时,需要同时维护前驱节点和后继节点的指针;单向链表只需维护一个方向的指针。
- 功能差异:双向链表可以快速访问前驱节点和后继节点,适用于需要双向遍历的场景;单向链表只能单向遍历,适用于只需单向访问的场景。
- 空间占用:双向链表由于每个节点有两个指针域,因此占用空间较大;单向链表占用空间较小。
- 实现复杂度:双向链表实现相对复杂,需要考虑前驱节点和后继节点的指针修改;单向链表实现简单。
在实际应用中,根据具体需求和场景选择合适的链表类型,可以提高程序的性能和效率。
热门推荐
皎月RQ连招中Ping值问题究竟如何影响游戏体验?
科学训练与饮食调整助你练出迷人马甲线的秘诀分享
绿色动能助推可持续交通
河北沧州:寻访大运河 探索“狮”文化
犯罪科心理学专家:剖析作案动机与行为模式的专业路径
人类历史上的“十二次”大规模屠杀事件,其中有6次发生在中国
高楼层每层都有风井吗?楼层选择不再迷茫,高楼层居住真相解析
马拉松达人分享:如何在一个月内为你的首次马拉松做准备?
王者荣耀直播:如何通过高超操作和战术分析吸引粉丝,打造火爆直播间
RCA(非平衡)与XLR(平衡)的区别
左侧肋骨神经痛症状是什么
A3尺寸对应的像素大小详解
专业logo设计手册制作指南:从基础到高级技巧全解析
90后管理者遇上80后下属:代际冲突背后的职场进化论
骆驼祥子第五章主要内容
韩国老师,被学生“霸凌”
水杉:中国特有珍稀植物的百科全书
火锅食材挑选指南:美味与健康兼得
“我的铁路风景”传递直抵人心的温暖与力量
回顾Telegram兴衰史:法律是重塑Web3应用的关键因素吗?
全球工业门类最齐全的国家,中国制造2025能超越欧美吗?
多空行情巧用 MACD 指标
如何识别传销:六大特征与防范策略
厉害的大人与孩子气的自己
应对宝宝拉肚子的有效方法与注意事项详解指南
三角梅的养殖方法和修剪
新闻热点预测:用数据建模洞察未来的社会脉搏
2025年春晚带火了新中式穿搭!王菲、赵雅芝等20位女明星穿搭解析
中国大陆儿童可以参加的含金量高的比赛有哪些?
羊为什么要磨牙?磨牙的作用是什么?