数据结构面试之二——双向链表表、循环链表、有序链表的常见操作

简介: 题注:《面试宝典》有相关习题,但思路相对不清晰,排版有错误,作者对此参考相关书籍和自己观点进行了重写,供大家参考。

二、双向链表

双向链表的建立是在单链表的基础上,多了一个指向前驱的指针back。其他的操作类似,注意点就是在双向链表的操作,尤其插入、删除操作中需要修改两个指针的指向,一个是back指针,一个是next指针。

1.双向链表的构建【前面插入】

构建双向链表注意点:1)修改first指针(头指针)的指向。2)修改back、next指针。

//反向表头插入,从前面插入...

template<typename Type>
nodeType<Type>*doublyLinkedList<Type>::buildListBackward()
{
       nodeType<Type> *newNode;
 
       int num;
       cout << " Enter a list of integer end with -999." << endl;
       cin >> num;
 
       while(num != -999)
       {
              //..add
              newNode = new nodeType<Type>;
              newNode->info = num;
              newNode->next = NULL;
              newNode->back = NULL;
             
              if(first == NULL)
              {
                     first = newNode;
              }
              else
              {
                     newNode->next = first;
                     first->back = newNode;
                     first = newNode;
              }
 
              cin >> num;
       }
       return first;
}

2.双向链表的插入节点【此处考虑插入后保证有序,直接插入排序】

节点的插入同单链表,依然需要考虑多种因素。分以下几类:

1)双向链表为空,“提示信息”并返回。

2)双向链表非空,待插入节点的元素值小于first节点,需要修改first指针。

3)双向链表非空,待插入节点的元素值为链表中间的节点,需修改back、next指向。

4)双向链表非空,待插入节点在最末尾节点以后,修改最末节点的指针。

//双向不循环链表
template<typename Type>
voiddoublyLinkedList<Type>::insertNode(const Type& insertItem)  //new
{
nodeType<Type> *current;
nodeType<Type> *trailCurrent;
nodeType<Type> *newNode;
bool found = false;
 
newNode = new nodeType<Type>;
newNode->info = insertItem;
newNode->next = NULL;
newNode->back = NULL;
 
//case 1: 空表
if(first == NULL)
{
        first = newNode;
}
else
{
        current = first;
        while( current != NULL)
        {
               if( current->info >= insertItem )
               {
                      found = true;
                      break;
               }
               else
               {
                      trailCurrent = current;
                      current = current->next;
               }
        }
 
        //case 2: 第一个节点
        if(current == first)
        {
               current->next = newNode;
               newNode->back = current;
        }
        else
        {
               //case 3: 中间节点
               if(current != NULL)
               {
                      newNode->next = trailCurrent->next;
                      current->back = newNode;
                      trailCurrent->next = newNode;
                      newNode->back = trailCurrent;
               }
               else  //case 4:最后一个节点前
               {
                      trailCurrent->next = newNode;
                      newNode->back = trailCurrent;
               }//end case 4
        }
}
}

3.双向链表的删除节点

同样需要考虑以下几点:

1) 链表为空,“提示信息”并返回。

2) 链表非空,查找删除的元素在链表是否存在。是的话,found=true;否的话,found=false。

3) 如果没有找到包含查找元素的节点,“错误提示”并返回。

4) 如果找到,主要节点位置。如果是头节点或末尾节点主要修改first指针,及节点的back、next指向;如果不是头节点或者末尾节点就是中间节点,主要back、next的指向,完成插入。

template<typename Type>
void doublyLinkedList<Type>::deleteNode(constType& deleteItem)   //new
{
  nodeType<Type> *current = new nodeType<Type>;
  nodeType<Type> *trailCurrent;
  bool found = false;
        
  //case 1: 空表
  if(first == NULL)
  {
         cout << "The List is NULL!" << endl;
         exit(1);
  }
  else
  {
         current = first;
         while( current != NULL)
         {
                if( current->info == deleteItem )
                {
                       found = true;
                       break;
                }
                else
                {
                       trailCurrent = current;
                       current = current->next;
                }
         }
 
         if(found)
         {
                //case 2: 第一个节点
                if(current == first)
                {
                       first = current->next;
                       delete current;
                }
                else
                {
                       //case 3: 中间节点
                       if(current != NULL)
                       {
                              if(current->next != NULL)  //case3.1:要删除的是中间节点.
                              {
                                     trailCurrent->next =current->next;
                                     current->next->back = trailCurrent;
                                     delete current;
                              }
                              else
                              {
                                     trailCurrent->next = NULL;//case3.2:要删除的为最后一个节点.
                                     delete current;
                              }
                       }
               
                }// end else
         }// end if
         else
         {
                cout << "The elem " <<deleteItem << " is not Exist in the List!" << endl;//case 4: 链表中无此节点.
         }
  }//end else
}//end

三、循环链表

循环链表能保障从每一个节点出发都能检索链表。即:last指针的指向不再为空,而是指向了first(头结点)。

循环链表在构建链表、查找元素、插入、删除、操作中要注意修改末尾节点指针的指向,保证链表的循环。

1.前插式构建循环链表

【思路】:每次在first指针前面插入节点;插入每个节点后注意修改last->link的指向。

template<typename Type>
void cycleList<Type>::bulidCycListBackward() //前插构造循环链表.
{
        Type newItem;
        while(cin >> newItem, newItem != -999)
        {
               nodeType<Type>* newNode = newnodeType<Type>;
               newNode->info = newItem;
               newNode->link = NULL;
 
               if(first == NULL)
               {
                      first = newNode;
                      last = newNode;
                      first->link = first;
                      last->link = first;
               }
               else
               {
                      newNode->link = first;
                      last->link = newNode;
                      first = newNode;
               }
        }
        cout << "输入完毕!" << endl;
}

2. 循环链表插入元素[后插入]

【思路】:同构造循环链表,每次在last指针后面插入节点;插入每个节点后注意修改last->link的指向。

//只在last末尾插入
template<typename Type>
voidcycleList<Type>::insertCycList(const Type& newItem)
{
nodeType<Type> *newNode = new nodeType<Type>;
newNode->info = newItem;
newNode->link = NULL;
 
if(first == NULL)        //链表为空...
{
        first = newNode;
        last = newNode;
        first->link = first;
        last->link = first;
}
else                   //链表非空...
{
        last->link = newNode;
        newNode->link = first;
        last = newNode;
}
cout << newItem << "was inserted!" <<endl;
}

3.删除循环链表节点

考虑到是否为空链表、删除元素在链表中不存在、及节点存在(节点的位置可能为共分头、中间、尾),所以分为以下5种情况分别处理。:

//case1:链表为空。
//case2:链表非空,删除节点为头节点。
//case3:链表非空,删除节点为尾节点。
//case4:链表非空,删除节点为中间节点。
//case5:链表非空,不存在=deleteItem的节点。
template<typename Type>
voidcycleList<Type>::delCycList(const Type& deleteItem)
{
nodeType<Type> *current = new nodeType<Type>;
nodeType<Type> *trailCurrent = new nodeType<Type>;
nodeType<Type> *temp ;
bool found = false;
 
if( first == NULL )
{
        cout << "The List is Empty!" << endl;//case1
        return;
}
if( first->info == deleteItem )      //case2
{
        temp = new nodeType<Type>;
        temp = first;
        first = first->link;
        last->link = first;
        cout << "The node has" << deleteItem<< " was deleted! " << endl;
        delete temp;
        return;
}
else                                             //cas3,case4.需要查找后定位。
{
        current = first->link;
        while( !found && current != first)
        {
               if( current->info == deleteItem )
               {
                      found = true;
                      break;
               }
               else
               {
                      trailCurrent = current;
                      current = current->link;
               }
        }// end while
 
        if(found)
        {    
               temp = new nodeType<Type>;
               if(current == last)                        //case3
               {
                      temp = last;
                      trailCurrent->link = last->link;       //last->link = first
                      last = trailCurrent;
               }
               else
               {
                      temp = current;
                     trailCurrent->link= current->link;
               }
               cout << "The Elem : " <<deleteItem << " was deleted! " << endl;
               delete temp;
        }
        else
        {
               cout << "The Elem : " <<deleteItem << " was not Exist in the List! " << endl;
        }
}//end else
}

四、有序链表

注意:此处有序链表无非是在之前的一、单链表的操作的基础上,在插入元素的时候,按顺序插入,考虑的排序。

为了体现有序的特点,特通过递归实现了有序链表的逆序打印。关于有序链表或者链表的非递归实现,可以通过栈实现。下一节会分析并实现。

1.有序链表的插入

考虑以下三种情况:

1) 当前链表为空;

2) 当前链表非空,要插入的元素值小于头结点的元素值;

3) 当前链表非空,要插入的元素值大于头结点的元素值,可以考虑找到第一个大于其的元素则停止搜索,插入其前即可。

template<typename Type>
void orderedLinkedListType<Type>::insertNode(constType& newItem)
{
nodeType<Type>* current;
nodeType<Type>* trailCurrent;
nodeType<Type>* newNode;
 
bool found = false;
 
newNode = new nodeType<Type>;
newNode->info = newItem;
newNode->link = NULL;
 
if(first == NULL)                           //case1:链表为空.
{
        first = newNode;
        last = first;
}
else
{
        current = first;                     
        while( !found && current != NULL)       //循环查找
        {
               if(current->info >= newItem)
               {
                      found = true;
                      break;
               }
              else
               {
                      trailCurrent = current;
                      current = current->link;
               }
        }//end while
 
        //first 特殊处理..
        if(current == first)                //case2:新插入的元素小于first节点的元素值..
        {
               newNode->link = first;         
               first = newNode;
        }
        else //其他..
        {
               newNode->link = current;       //case3:新插入的节点在非first位置
               trailCurrent->link = newNode;
        }
}//end else
}

2.      递归实现有序链表的逆序打印

template<typename Type>
voidorderedLinkedListType<Type>::printListReverse() const   //逆序打印.
{
     reversePrint(first);
     cout << endl;
}
 
//递归实现单链表的逆序打印.
template<typename Type>
voidorderedLinkedListType<Type>::reversePrint(nodeType<Type>* current)const  //逆序打印
{
if(current != NULL)
{
        reversePrint(current->link);
        cout << current->info << "\t";
}
}

作者:铭毅天下

原文:https://blog.csdn.net/laoyang360/article/details/7866886

相关文章
|
10天前
|
存储 Java 索引
【数据结构】链表从实现到应用,保姆级攻略
本文详细介绍了链表这一重要数据结构。链表与数组不同,其元素在内存中非连续分布,通过指针连接。Java中链表常用于需动态添加或删除元素的场景。文章首先解释了单向链表的基本概念,包括节点定义及各种操作如插入、删除等的实现方法。随后介绍了双向链表,说明了其拥有前后两个指针的特点,并展示了相关操作的代码实现。最后,对比了ArrayList与LinkedList的不同之处,包括它们底层实现、时间复杂度以及适用场景等方面。
31 10
【数据结构】链表从实现到应用,保姆级攻略
|
28天前
|
存储 算法
【初阶数据结构篇】顺序表和链表算法题
此题可以先找到中间节点,然后把后半部分逆置,最近前后两部分一一比对,如果节点的值全部相同,则即为回文。
|
28天前
|
存储 测试技术
【初阶数据结构篇】双向链表的实现(赋源码)
因为头结点的存在,plist指针始终指向头结点,不会改变。
|
28天前
|
存储 测试技术
【初阶数据结构篇】单链表的实现(附源码)
在尾插/尾删中,都需要依据链表是否为空/链表是否多于一个节点来分情况讨论,目的是避免对空指针进行解引用造成的错误。
|
1月前
|
算法
【数据结构与算法】共享双向链表
【数据结构与算法】共享双向链表
10 0
|
1月前
|
算法
【数据结构与算法】双向链表
【数据结构与算法】双向链表
10 0
|
4天前
|
存储 人工智能 C语言
数据结构基础详解(C语言): 栈的括号匹配(实战)与栈的表达式求值&&特殊矩阵的压缩存储
本文首先介绍了栈的应用之一——括号匹配,利用栈的特性实现左右括号的匹配检测。接着详细描述了南京理工大学的一道编程题,要求判断输入字符串中的括号是否正确匹配,并给出了完整的代码示例。此外,还探讨了栈在表达式求值中的应用,包括中缀、后缀和前缀表达式的转换与计算方法。最后,文章介绍了矩阵的压缩存储技术,涵盖对称矩阵、三角矩阵及稀疏矩阵的不同压缩存储策略,提高存储效率。
|
6天前
|
存储 C语言
数据结构基础详解(C语言): 栈与队列的详解附完整代码
栈是一种仅允许在一端进行插入和删除操作的线性表,常用于解决括号匹配、函数调用等问题。栈分为顺序栈和链栈,顺序栈使用数组存储,链栈基于单链表实现。栈的主要操作包括初始化、销毁、入栈、出栈等。栈的应用广泛,如表达式求值、递归等场景。栈的顺序存储结构由数组和栈顶指针构成,链栈则基于单链表的头插法实现。
|
8天前
|
Java
【数据结构】栈和队列的深度探索,从实现到应用详解
本文介绍了栈和队列这两种数据结构。栈是一种后进先出(LIFO)的数据结构,元素只能从栈顶进行插入和删除。栈的基本操作包括压栈、出栈、获取栈顶元素、判断是否为空及获取栈的大小。栈可以通过数组或链表实现,并可用于将递归转化为循环。队列则是一种先进先出(FIFO)的数据结构,元素只能从队尾插入,从队首移除。队列的基本操作包括入队、出队、获取队首元素、判断是否为空及获取队列大小。队列可通过双向链表或数组实现。此外,双端队列(Deque)支持两端插入和删除元素,提供了更丰富的操作。
11 0
【数据结构】栈和队列的深度探索,从实现到应用详解
|
1月前
栈的几个经典应用,真的绝了
文章总结了栈的几个经典应用场景,包括使用两个栈来实现队列的功能以及利用栈进行对称匹配,并通过LeetCode上的题目示例展示了栈在实际问题中的应用。
栈的几个经典应用,真的绝了