深入解析双向链表与单向链表的区别:示例详解
创作时间:
作者:
@小白创作中心
深入解析双向链表与单向链表的区别:示例详解
引用
CSDN
1.
https://blog.csdn.net/qq_35320456/article/details/140401203
链表是一种灵活的数据结构,它通过指针连接一系列节点,实现了动态数组的特性。在众多链表类型中,单向链表和双向链表是最为常见的两种。本文将重点探讨这两种链表的区别,并通过示例进行详细说明。
一、单向链表与双向链表的定义及结构
单向链表
定义:单向链表是链表的一种,每个节点包含数据域和一个指向下一个节点的指针。
结构示例:
节点Node {
int data; // 数据域
Node next; // 指针域,指向下一个节点
}
双向链表
定义:双向链表是链表的一种,每个节点包含数据域、一个指向前一个节点的指针和一个指向下一个节点的指针。
结构示例:
节点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);
ist.insertAtHead(2);
list.insertAtHead(1);
list.printList();
return 0;
}
四、总结
通过以上示例,我们可以总结出单向链表与双向链表以下几个主要区别:
- 结构差异:双向链表每个节点有两个指针域,分别指向前一个节点和后一个节点;单向链表只有一个指针域,指向下一个节点。
- 操作差异:双向链表在插入和删除操作时,需要同时维护前驱节点和后继节点的指针;单向链表只需维护一个方向的指针。
- 功能差异:双向链表可以快速访问前驱节点和后继节点,适用于需要双向遍历的场景;单向链表只能单向遍历,适用于只需单向访问的场景。
- 空间占用:双向链表由于每个节点有两个指针域,因此占用空间较大;单向链表占用空间较小。
- 实现复杂度:双向链表实现相对复杂,需要考虑前驱节点和后继节点的指针修改;单向链表实现简单。
在实际应用中,根据具体需求和场景选择合适的链表类型,可以提高程序的性能和效率。
热门推荐
有机农业的四大原则及其对环境和人类健康的影响

从“观光”到“体验” AI助力文旅“黑骏马”
足疗店人身安全合同的法律制定与风险防范
郦道元《三峡》:巴东三峡巫峡长,猿鸣三声泪沾裳
打造成功创业团队的五大关键要素分析
如何根据个人兴趣和市场需求选择创业行业?
毒螃蟹来历揭秘:环境污染、食物链紊乱等多重因素导致
监控录像机硬盘怎么查看通电
网贷逾期用私人号码催收会怎么样
保持低盐低糖低脂 低蛋白饮食对肾功能好
为什么医生看到这种“痣”建议查查肝?肝不好真的可能写在脸上
名媛如何通过多元化的途径实现财富积累和收入增长
DOGE宣称节省数百亿,实际数据引发质疑
电动车电池加注蒸馏水的完整步骤
本科生考试不合格率超过30%,老师应该承担责任吗?
辐射4哪个是好的势力
如何在市场波动中调整投资策略?这种调整对风险管理有何影响?
日本刑事案件连续三年上升,背后的社会隐忧值得关注
初高中周末双休、取消晚自习成大趋势,网友吵翻:减负还是依然卷?
工业铝合金在现代制造业的广泛应用
美国有哪些银行比较有名?——全面了解美国顶级银行
电子信息工程和通信工程哪个好就业、前景更好?附全方面对比
电子信息工程和通信工程哪个好就业、前景更好?附全方面对比
室内燃气管道安装规范
荷叶山楂茶的健益处与禁忌人群:一份详细的指南
小小垃圾桶引领乡村新风尚
电子产品手板的小批量生产,CNC加工是否降低库存压力?
怎么使存货减少?这些方法及其局限性你需要了解
牙龈出血要治吗?劝你早去看医生
成都出发自驾游康定,探索川西美景