数据结构单链表之链表插入 | 第三套

简介: 数据结构单链表之链表插入 | 第三套

我们在上一篇文章中介绍了链表。还创建了一个具有 3 个节点的简单链表并讨论了链表遍历。 本文中讨论的所有程序都考虑以下链表表示。

// 一个链表节点
class Node
{
  public:
  int data;
  Node *next;
};

在这篇文章中,讨论了在链表中插入新节点的方法。可以通过三种方式添加节点

1) 在链表的前面

2) 在给定节点之后。

3) 在链表的末尾。

在前面添加一个节点:(4 步过程)  

新节点总是添加在给定链表的头部之前。并且新添加的节点成为链表的新头。例如,如果给定的链表是 10->15->20->25 并且我们在前面添加了一个项目 5,那么链表就变成了 5->10->15->20->25。让我们调用添加在列表前面的函数是 push()。push() 必须接收一个指向头指针的指针,因为 push 必须将头指针更改为指向新节点

image.png

以下是在前面添加节点的 4 个步骤。

/* 给定一个指向列表头部的引用(指向指针的指针)和一个 int,在列表的前面插入一个新节点。 */
void push(Node** head_ref, int new_data)
{
  /* 1. 分配节点 */
  Node* new_node = new Node();
  /* 2. 放入数据 */
  new_node->data = new_data;
  /* 3. 将新节点的下一个作为头 */
  new_node->next = (*head_ref);
  /* 4. 移动头部指向新节点 */
  (*head_ref) = new_node;
}

push() 的时间复杂度是 O(1),因为它做的工作量是恒定的。

在给定节点之后添加一个节点:(5 个步骤过程)  

我们得到一个指向节点的指针,在给定节点之后插入新节点。

image.png

// 给定一个节点 prev_node,在给定的 prev_node 后面插入一个新节点
void insertAfter(Node* prev_node, int new_data)
{
  // 1. 检查给定的 prev_node 是否为 NULL if (prev_node == NULL)
  {
    cout << "给定的前一个节点不能为 NULL";
    return;
  }
  // 2. 分配新节点
  Node* new_node = new Node();
  // 3.放入数据
  new_node->data = new_data;
  // 4. 将新节点的下一个设为 prev_node 的下一个
  new_node->next = prev_node->next;
  // 5.将 prev_node 的下一个移动为 new_node
  prev_node->next = new_node;
}

insertAfter() 的时间复杂度是 O(1),因为它做的工作量是恒定的。

最后添加一个节点:(6 步过程)

新节点总是添加在给定链表的最后一个节点之后。例如,如果给定的链表是 5->10->15->20->25 并且我们在最后添加了一个项目 30,那么链表变成了 5->10->15->20->25- > 30。

由于链表通常由它的头部表示,我们必须遍历链表直到最后,然后将倒数第二个节点更改为新节点。

image.png

以下是最后添加节点的 6 个步骤。

c++

复制代码

// 给定一个指向列表头部的引用(指向指针的指针)和一个 int,在末尾附加一个新节点
void append(Node** head_ref, int new_data)
{
  // 1. 分配节点
  Node* new_node = new Node();
  //在步骤 5 中使用
  Node *last = *head_ref;
  // 2. 放入数据
  new_node->data = new_data;
  // 3. 这个新节点将是最后一个节点,所以将它的 next 设为 NULL
  new_node->next = NULL;
  // 4. 如果链表为空,则以新节点为头
  if (*head_ref == NULL)
  {
    *head_ref = new_node;
    return;
  }
  // 5. else 遍历到最后一个节点
  while (last->next != NULL)
    last = last->next;
  // 6. 更改最后一个节点的下一个
  last->next = new_node;
  return;
}

append 的时间复杂度为 O(n),其中 n 是链表中的节点数。由于从头到尾有一个循环,该函数执行 O(n) 工作。

通过保留一个指向链表尾部的额外指针/

下面是一个完整的程序,它使用上述所有方法来创建一个链表。

#include <bits/stdc++.h>
using namespace std;
class Node
{
  public:
  int data;
  Node *next;
};
void push(Node** head_ref, int new_data)
{
  Node* new_node = new Node();
  new_node->data = new_data;
  new_node->next = (*head_ref);
  (*head_ref) = new_node;
}
void insertAfter(Node* prev_node, int new_data)
{
  if (prev_node == NULL)
  {
    cout<<"给定的前一个节点不能为 NULL";
    return;
  }
  Node* new_node = new Node();
  new_node->data = new_data;
  new_node->next = prev_node->next;
  prev_node->next = new_node;
}
void append(Node** head_ref, int new_data)
{
  Node* new_node = new Node();
  Node *last = *head_ref; 
  new_node->data = new_data;
  new_node->next = NULL;
  if (*head_ref == NULL)
  {
    *head_ref = new_node;
    return;
  }
  while (last->next != NULL)
    last = last->next;
  last->next = new_node;
  return;
}
void printList(Node *node)
{
  while (node != NULL)
  {
    cout<<" "<<node->data;
    node = node->next;
  }
}
int main()
{
  Node* head = NULL;
  append(&head, 6);
  push(&head, 7);
  push(&head, 1);
  append(&head, 4);
  insertAfter(head->next, 8);
  cout<<"创建的链表是:";
  printList(head);
  return 0;
}

输出:

 创建的链表是:1 7 8 6 4


目录
相关文章
|
21天前
|
存储 C语言
【数据结构】手把手教你单链表(c语言)(附源码)
本文介绍了单链表的基本概念、结构定义及其实现方法。单链表是一种内存地址不连续但逻辑顺序连续的数据结构,每个节点包含数据域和指针域。文章详细讲解了单链表的常见操作,如头插、尾插、头删、尾删、查找、指定位置插入和删除等,并提供了完整的C语言代码示例。通过学习单链表,可以更好地理解数据结构的底层逻辑,提高编程能力。
48 4
|
1月前
|
算法 程序员 索引
数据结构与算法学习七:栈、数组模拟栈、单链表模拟栈、栈应用实例 实现 综合计算器
栈的基本概念、应用场景以及如何使用数组和单链表模拟栈,并展示了如何利用栈和中缀表达式实现一个综合计算器。
31 1
数据结构与算法学习七:栈、数组模拟栈、单链表模拟栈、栈应用实例 实现 综合计算器
|
22天前
|
算法 安全 搜索推荐
2024重生之回溯数据结构与算法系列学习之单双链表精题详解(9)【无论是王道考研人还是IKUN都能包会的;不然别给我家鸽鸽丢脸好嘛?】
数据结构王道第2.3章之IKUN和I原达人之数据结构与算法系列学习x单双链表精题详解、数据结构、C++、排序算法、java、动态规划你个小黑子;这都学不会;能不能不要给我家鸽鸽丢脸啊~除了会黑我家鸽鸽还会干嘛?!!!
|
22天前
|
存储 Web App开发 算法
2024重生之回溯数据结构与算法系列学习之单双链表【无论是王道考研人还是IKUN都能包会的;不然别给我家鸽鸽丢脸好嘛?】
数据结构之单双链表按位、值查找;[前后]插入;删除指定节点;求表长、静态链表等代码及具体思路详解步骤;举例说明、注意点及常见报错问题所对应的解决方法
|
1月前
|
存储
[数据结构] -- 单链表
[数据结构] -- 单链表
26 1
|
1月前
|
算法 Java
数据结构与算法学习五:双链表的增、删、改、查
双链表的增、删、改、查操作及其Java实现,并通过实例演示了双向链表的优势和应用。
17 0
数据结构与算法学习五:双链表的增、删、改、查
|
21天前
|
C语言
【数据结构】双向带头循环链表(c语言)(附源码)
本文介绍了双向带头循环链表的概念和实现。双向带头循环链表具有三个关键点:双向、带头和循环。与单链表相比,它的头插、尾插、头删、尾删等操作的时间复杂度均为O(1),提高了运行效率。文章详细讲解了链表的结构定义、方法声明和实现,包括创建新节点、初始化、打印、判断是否为空、插入和删除节点等操作。最后提供了完整的代码示例。
39 0
|
1月前
|
存储
[数据结构] -- 双向循环链表
[数据结构] -- 双向循环链表
23 0
|
1月前
|
存储
数据结构(单链表)
数据结构(单链表)
16 0
|
1月前
|
存储
探索数据结构:便捷的双向链表
探索数据结构:便捷的双向链表

热门文章

最新文章

下一篇
无影云桌面