数据结构— —双向链表

简介: 数据结构— —双向链表

数据结构— —双向链表


双向链表的算法实现


单链表中每个结点除了存储自身数据之后,还存储了下一个结点的地址,因此可以轻松访问下一个结点,以及后面的后继结点,但是如果想访问前面的结点就不行了,再也回不去了。例如删除结点 p 时,要先找到它的前一个结点 q,然后才能删掉 p 结点,单向链表只能往后走,不能向前走。如果需要向前走,怎么办呢?


可以在单链表的基础上给每个元素附加两个指针域,一个存储前一个元素的地址,一个存储下一个元素的地址。这种链表称为双向链表.


91db747eef804b0daa40d81a1059b5ac.png

其结构体定义:

typedef struct _LinkNode {


int data; //结点的数据域 struct _LinkNode *next; //下一个节点的指针域


struct _LinkNode *prev; //上一个结点的指针域


}LinkNode, LinkList; //LinkList 为指向结构体 LNode 的指针类型


双向链表的初始


//前插法
bool DbListInsert_front(DbLinkList* &L, DbLinkNode *node){ if(!L || !node) return false;
//1.只有头节点
if(L->next==NULL){
typedef struct _DoubleLinkNode { int data; //结点的数据域
struct _DoubleLinkNode *next; //下一个节点的指针域 struct _DoubleLinkNode *prev; //上一个结点的指针域
}DbLinkNode, DbLinkList; //LinkList为指向结构体LNode的指针类型
bool DbInit_List(DbLinkList* &L)//构造一个空的双向链表L
{
L=new DbLinkNode; //生成新结点作为头结点,用头指针L指向头结点 if(!L)return false; //生成结点失败
L->next=NULL; //头结点的next指针域置空
L->prev=NULL; //头结点的指针域置空 L->data = -1; return true;
}

双向链表增加元素前插法

 node->next=NULL;                           11
node->prev=L; //新节点prev 指针指向头节点 L->next=node; //头节点next 指针指向新节点
}else {
L->next->prev=node;  //第二个节点的prev 指向新节点 node->next = L->next; //新节点next指针指向第二个节点 node->prev=L;    //新节点prev 指针指向头节点
                  L->next=node;            //头节点next 指针指向新节点,完成插入
}
return true;
}


0617f37fffae41f6bffff9516cb606a6.png

尾插法

//尾插法
bool DbListInsert_back(DbLinkList* &L, DbLinkNode *node){ DbLinkNode *last = NULL; if(!L || !node) return false;
last = L;
while(last->next) last = last->next;
node->next = NULL;
last->next = node;
node->prev = last; return true; }

d5f4364e7465468f80aa4ca7a80b6ecb.png

任意位置插入

//指定位置插入
bool DbLink_Insert(DbLinkList* &L, int i, int &e){ if(!L||!L->next) return false; if(i<1) return false;
int j =0;
DbLinkList *p, *s;
p = L;
while(p && j<i){//查找位置为i的结点,p指向该结点 p = p->next;
j++;
}
if(!p || j!=i){ cout<<"不存在节点:"<<i<<endl;
return false;
} cout<<"p: "<<p<<endl;
s=new DbLinkNode;//生成新节点
s->data = e;
s->next = p;
s->prev = p->prev;
p->prev->next = s;
p->prev = s;
return true;
}

双向链表的遍历

//双向链表的遍历输出
void DbLink_Print(DbLinkList* &L ){
DbLinkNode *p = NULL;
if(!L){
cout<<"链表为空."<<endl; return ;
}
p = L;
while(p->next){
cout<<p->next->data<<"\t";
p = p->next;
}
//逆向打印
cout<<endl<<"逆向打印"<<endl;
while(p){
cout<<p->data<<"\t";
p = p->prev;
}
cout<<endl;
}

双向链表获取元素

bool DbLink_GetElem(DbLinkList* &L, int i, int &e)//双向链表的取值
{
//在带头结点的双向链表L中查找第i个元素
//用e记录L中第i个数据元素的值
int index;
DbLinkList *p;
if(!L || !L->next) return false;
p = L->next;
index = 1;
while(p && index<i){//顺链表向后扫描,直到p指向第i个元素或p为空
p = p->next;
index++;
}
if(!p || index>i){ return false;  //i值不合法,i>n或i<=0
}
e=p->data; return true;
}


11

双向链表删除元素

//任意位置删除
bool DbLink_Delete(DbLinkList* &L, int i) //双向链表的删除
{
DbLinkList *p;
int index = 0;
if(!L || !L->next){
cout<<"双向链表为空!"<<endl;
return false;
}
if(i<1) return false; //不能删除头节点
p=L;
while(p && index<i){
p = p->next;
index++;
}
if(!p){ //当节点不存在时,返回失败 return false;
}
p->prev->next=p->next; //改变删除结点前驱结点的next 指针域 if(p->next){
p->next->prev = p->prev; //改变删除节点后继节点的prev 指针域
}
delete p;         //释放被删除结点的空间
return true;
}

双向链表销毁

void DbLink_Destroy(DbLinkList* &L) //双向链表的销毁
{
//定义临时节点p指向头节点
DbLinkList *p = L;
cout<<"销毁链表!"<<endl;
while(p){
L=L->next;//L指向下一个节点 cout<<"删除元素: "<<p->data<<endl; delete p; //删除当前节点
p = L;   //p 移向下一个节点
}
}

完整代码实现

#include <iostream>
#include <Windows.h>
using namespace std;
typedef struct _DbLinkList
{
  int data;//数据域
  struct _DbLinkList* prev;//前驱结点
  struct _DbLinkList* next;//后继结点
}DbLinkList, DbLinkNode;
bool DbLinkList_Init(DbLinkList*& L)
{
  //分配头结点
  L = new DbLinkNode;
  if (!L)
  {
    return false;
  }
  L->next = NULL;
  L->prev = NULL;
  L->data = -1;
  return true;
}
//头插
bool InsertDbLink_front(DbLinkList*& L, DbLinkList* node)
{
  if (!L || !node)
  {
    return false;
  }
  if (L->next == NULL)
  {
    node->next = NULL;
    node->prev = L;
    L->next = node;
  }
  else
  {
    L->next->prev = node;
    node->next = L->next;
    node->prev = L;
    L->next = node;
    //L->next->prev = node; //第二个节点的 prev 指向新节点
    //node->next = L->next; //新节点 next 指针指向第二个节点
    //node->prev = L; //新节点 prev 指针指向头节点
    //L->next = node;
  }
  return true;
}
//尾插
bool InsertDbLink_back(DbLinkList* &L, DbLinkNode* node)
{
  if (!L || !node)
  {
    return false;
  }
  DbLinkList* q;
  q = L;
  while (q->next)
  {
    q = q->next;
  }
  q->next = node;
  node->prev = node;
  return true;
}
//任意位置插入
bool DbLink_Insert(DbLinkList*& L,int i,int &e)
{
  DbLinkList* q, * s;
  int j = 1;
  q = L->next;
  if (!L || !L->next)
  {
    return false;
  }
  while (!q || j < i)
  {
    q = q->next;
    j++;
  }
  if (!q || i != j)
  {
    return false;
  }
  s = new DbLinkNode;
  s->data = e;
  q->prev->next = s;
  s->prev = q->prev;
  s->next = q;
  q->prev = s;
  return true;
}
//按指定位置查找元素
bool seekElem(DbLinkList*& L, int i, int& e)
{
  DbLinkList* p;
  int j = 1;
  p = L->next;
  if (!L || !L->next)
  {
    return false;
  }
  while (p && j < i)
  {
    p = p->next;
    j++;
  }
  if (!p || j > i)
  {
    return false;
  }
  e = p->data;
  return true;
}
bool DbLink_Delete(DbLinkList*& L, int i)
{
  DbLinkNode* pos;
  int j = 1;
  pos = L->next;
  if (!L || !L->next)
  {
    return false;
  }
  if (i < 1)
  {
    return false;
  }
  while (!pos ||j<i)
  {
    pos = pos->next;
    j++;
  }
  if (!pos)
  {
    return false;
  }
  pos->prev->next = pos->next;
  pos->next->prev = pos->prev;
  delete pos;
  return true;
}
//链表的销毁
void DestroyLink(DbLinkList* &L)
{
  DbLinkNode* q = L;
  if (!L)
  {
    cout << "链表为空" << endl;
  }
  while (q)
  {
    L = L->next;
    delete q;
    q = L;
  }
  cout << "链表销毁了" << endl;
}
void DbLink_Print(DbLinkList*& L)
{
  DbLinkNode* s = NULL;
  s = L;
  if (!L)
  {
    cout << "链表为空." << endl;
    return;
  }
  while (s->next)
  {
    s = s->next;
    cout << s->data << "\t";
  }
  cout << endl;
  while (s!=L)
  {
    cout << s->data << "\t";
    s = s->prev;
  }
  cout << endl;
}
int main(void)
{
  DbLinkList* L, * s, * a;
  //双向链表初始化
  if (DbLinkList_Init(L))
  {
    cout << "初始化循环链表成功" << endl;
  }
  else
  {
    cout << "初始化循环链表失败" << endl;
  }
  //前插
  int j;
  cout << "请输入你要插入的个数:";
  cin >> j;
  for (int i = 0; i < j; i++)
  {
    s = new DbLinkNode;
    cin >> s->data;
    InsertDbLink_front(L, s);
  }
  DbLink_Print(L);
  //尾插
  int z;
  cout << "请输入你要插入的个数:";
  cin >> z;
  for (int i = 0; i < z; i++)
  {
    a = new DbLinkNode;
    cin >> a->data;
    InsertDbLink_front(L, a);
  }
  DbLink_Print(L);
  //任意位置插入
  int element, w;
  cout << "请输入你要插入的数:";
  cin >> w;
  cout << "请输入你要插入的位置:";
  cin >> element;
  DbLink_Insert(L, element, w);
  DbLink_Print(L);
  //查找元素
  int b;
  cout << "请输入你要查找元素的位置:";
  cin >> b;
  seekElem(L, b, element);
  cout << element << endl;
  //删除指定位置的元素
  DbLink_Delete(L, 3);
  DbLink_Print(L);
  //销毁链表
  DestroyLink(L);
  DbLink_Print(L);
  system("pause");
  return 0;
}


相关文章
|
2月前
|
存储 算法 Perl
数据结构实验之链表
本实验旨在掌握线性表中元素的前驱、后续概念及链表的建立、插入、删除等算法,并分析时间复杂度,理解链表特点。实验内容包括循环链表应用(约瑟夫回环问题)、删除单链表中重复节点及双向循环链表的设计与实现。通过编程实践,加深对链表数据结构的理解和应用能力。
71 4
|
11天前
|
机器学习/深度学习 存储 C++
【C++数据结构——线性表】单链表的基本运算(头歌实践教学平台习题)【合集】
本内容介绍了单链表的基本运算任务,涵盖线性表的基本概念、初始化、销毁、判定是否为空表、求长度、输出、求元素值、按元素值查找、插入和删除数据元素等操作。通过C++代码示例详细解释了顺序表和链表的实现方法,并提供了测试说明、通 - **任务描述**:实现单链表的基本运算。 - **相关知识**:包括线性表的概念、初始化、销毁、判断空表、求长度、输出、求元素值、查找、插入和删除等操作。 - **测试说明**:平台会对你编写的代码进行测试,提供测试输入和预期输出。 - **通关代码**:给出了完整的C++代码实现。 - **测试结果**:展示了测试通过后的预期输出结果。 开始你的任务吧,祝你成功!
30 5
|
25天前
|
数据库
数据结构中二叉树,哈希表,顺序表,链表的比较补充
二叉搜索树,哈希表,顺序表,链表的特点的比较
数据结构中二叉树,哈希表,顺序表,链表的比较补充
|
2月前
|
存储 缓存 算法
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式,强调了合理选择数据结构的重要性,并通过案例分析展示了其在实际项目中的应用,旨在帮助读者提升编程能力。
87 5
|
2月前
|
存储 C语言
【数据结构】手把手教你单链表(c语言)(附源码)
本文介绍了单链表的基本概念、结构定义及其实现方法。单链表是一种内存地址不连续但逻辑顺序连续的数据结构,每个节点包含数据域和指针域。文章详细讲解了单链表的常见操作,如头插、尾插、头删、尾删、查找、指定位置插入和删除等,并提供了完整的C语言代码示例。通过学习单链表,可以更好地理解数据结构的底层逻辑,提高编程能力。
137 4
|
2月前
|
算法 安全 搜索推荐
2024重生之回溯数据结构与算法系列学习之单双链表精题详解(9)【无论是王道考研人还是IKUN都能包会的;不然别给我家鸽鸽丢脸好嘛?】
数据结构王道第2.3章之IKUN和I原达人之数据结构与算法系列学习x单双链表精题详解、数据结构、C++、排序算法、java、动态规划你个小黑子;这都学不会;能不能不要给我家鸽鸽丢脸啊~除了会黑我家鸽鸽还会干嘛?!!!
|
2月前
|
存储 Web App开发 算法
2024重生之回溯数据结构与算法系列学习之单双链表【无论是王道考研人还是IKUN都能包会的;不然别给我家鸽鸽丢脸好嘛?】
数据结构之单双链表按位、值查找;[前后]插入;删除指定节点;求表长、静态链表等代码及具体思路详解步骤;举例说明、注意点及常见报错问题所对应的解决方法
|
2月前
|
算法
数据结构之购物车系统(链表和栈)
本文介绍了基于链表和栈的购物车系统的设计与实现。该系统通过命令行界面提供商品管理、购物车查看、结算等功能,支持用户便捷地管理购物清单。核心代码定义了商品、购物车商品节点和购物车的数据结构,并实现了添加、删除商品、查看购物车内容及结算等操作。算法分析显示,系统在处理小规模购物车时表现良好,但在大规模购物车操作下可能存在性能瓶颈。
61 0
|
3月前
|
存储 Java
数据结构第三篇【链表的相关知识点一及在线OJ习题】
数据结构第三篇【链表的相关知识点一及在线OJ习题】
38 7
|
3月前
|
存储 安全 Java
【用Java学习数据结构系列】探索顺序表和链表的无尽秘密(附带练习唔)pro
【用Java学习数据结构系列】探索顺序表和链表的无尽秘密(附带练习唔)pro
32 3

热门文章

最新文章