HOW - 链表系列(一)

ops/2024/11/14 0:12:57/

目录

  • 一、JavaScript 中的链表
  • 二、链表
    • 移除链表元素
    • 反转链表
      • 1. 双指针法
      • 2. 递归法1:从前往后翻转指针指向
      • 3. 递归法2:从后往前翻转指针指向

一、JavaScript 中的链表

在JavaScript中,链表并不像在某些其他编程语言中那样常见。这是因为JavaScript中的数组(Array)数据结构通常能够满足大多数数据存储和操作的需求,而且更加方便和易用。

以下是一些可能导致JavaScript中较少使用链表的原因:

  1. 数组的灵活性:JavaScript中的数组是动态的,并且可以根据需要自动调整大小。这使得数组在存储和访问元素时非常方便,并且支持随机访问,即可以通过索引直接访问任意位置的元素。

  2. 内存管理:JavaScript中的内存管理是由垃圾收集器(Garbage Collector)自动处理的,它负责分配和释放内存。而链表在插入和删除节点时需要手动管理内存,需要考虑节点的创建和销毁,更容易出现内存泄漏等问题。

  3. 缺乏指针操作:JavaScript是一种高级语言,它的设计目标是提供简单易用的编程体验。与低级语言相比,JavaScript缺乏直接的指针操作,这使得链表的实现和操作变得相对复杂和笨重。

尽管如此,链表在一些特定的应用场景下仍然有其优势。例如,当需要频繁进行插入和删除操作而不关心随机访问时,链表可能更适合。此外,一些算法和数据结构问题也需要使用链表进行解决。

虽然JavaScript中链表的使用相对较少,但了解链表的概念和实现方式仍然是有益的,因为它们在计算机科学中是一种重要的数据结构,并广泛应用于其他编程语言和算法领域。

二、链表

关于链表的基础内容这里不进行介绍,我们将通过一些经典场景来学习和掌握链表相关操作。

移除链表元素

删除链表中等于给定值 val 的所有节点。

示例 1: 输入:head = [1,2,6,3,4,5,6], val = 6 输出:[1,2,3,4,5]

示例 2: 输入:head = [], val = 1 输出:[]

示例 3: 输入:head = [7,7,7,7], val = 7 输出:[]

// 定义链表节点类
class ListNode {constructor(val, next) {this.val = val;this.next = next || null;}
}
// 创建链表
const head = new ListNode(1);
let node2 = new ListNode(2);
let node3 = new ListNode(3);
let node4 = new ListNode(4);
let node5 = new ListNode(5);
head.next = node2;
node2.next = node3;
node3.next = node4;
node4.next = node5;var removeElements = function(head, val) {// 推荐定义一个额外的虚拟头节点,可以保证真实头节点和后续的普通节点用同一个操作逻辑const ret = new ListNode(0, head);let cur = ret;while(cur.next) {if(cur.next.val === val) {cur.next =  cur.next.next;continue;}cur = cur.next;}return ret.next;
};
removeElements(head, 4);
console.log(head);

反转链表

反转一个单链表

示例: 输入: 1->2->3->4->5->NULL 输出: 5->4->3->2->1->NULL

如果再定义一个新的链表,实现链表元素的反转,其实这是对内存空间的浪费。其实只需要改变链表的 next 指针的指向,直接将链表反转 ,而不用重新定义一个新的链表

1. 双指针法

思路:首先定义一个 cur 指针,指向头结点,再定义一个 pre 指针,初始化为 null。

然后就要开始反转了,首先要把 cur->next 节点用 tmp 指针保存一下,也就是保存一下这个节点。为什么要保存一下这个节点呢,因为接下来要改变 cur->next 的指向了,将 cur->next 指向pre ,此时已经反转了第一个节点了。

接下来,就是循环走如下代码逻辑了,继续移动 pre 和 cur 指针。

最后,cur 指针已经指向了null,循环结束,链表也反转完毕了。 此时我们 return pre 指针就可以了,pre 指针就指向了新的头结点。

// 定义链表节点类
class ListNode {constructor(val, next) {this.val = val;this.next = next || null;}
}
// 创建链表
const head = new ListNode(1);
let node2 = new ListNode(2);
let node3 = new ListNode(3);
let node4 = new ListNode(4);
let node5 = new ListNode(5);
head.next = node2;
node2.next = node3;
node3.next = node4;
node4.next = node5;// 反转链表
let newHead = reverseLinkedList(head);
function reverseLinkedList(head) {let pre = null;let cur = head;while (cur) {let temp = cur.next;cur.next = pre;pre = cur;cur = temp;}return pre;
}console.log(newHead);

时间复杂度: O(n),空间复杂度: O(1)。

2. 递归法1:从前往后翻转指针指向

递归法相对抽象一些,但是其实和双指针法是一样的逻辑,同样是当 cur 为空的时候循环结束,不断将 cur 指向 pre 的过程。关键是初始化的地方,从下方代码实现可以看到双指针法中初始化 cur = head, pre = NULL,在递归法中初始化的逻辑也是一样的,只不过写法变了。

// 定义链表节点类
class ListNode {constructor(val, next) {this.val = val;this.next = next || null;}
}
// 创建链表
const head = new ListNode(1);
let node2 = new ListNode(2);
let node3 = new ListNode(3);
let node4 = new ListNode(4);
let node5 = new ListNode(5);
head.next = node2;
node2.next = node3;
node3.next = node4;
node4.next = node5;// 反转链表
var reverse = function(pre, head) {if(!head) return pre;const temp = head.next;head.next = pre;pre = head;return reverse(pre, temp); // cur=temp; pre=cur;
}
var reverseLinkedList = function(head) {return reverse(null, head); // 初始化
};
const newHead = reverseLinkedList(head);console.log(newHead);

时间复杂度: O(n), 要递归处理链表的每个节点,空间复杂度: O(n), 递归调用了 n 层栈空间。

3. 递归法2:从后往前翻转指针指向

// 定义链表节点类
class ListNode {constructor(val, next) {this.val = val;this.next = next || null;}
}
// 创建链表
const head = new ListNode(1);
let node2 = new ListNode(2);
let node3 = new ListNode(3);
let node4 = new ListNode(4);
let node5 = new ListNode(5);
head.next = node2;
node2.next = node3;
node3.next = node4;
node4.next = node5;// 反转链表
var reverse = function(head) {if(!head || !head.next) return head;// 从后往前翻console.log(head.val);const pre = reverse(head.next);console.log(pre.val);head.next = pre.next;pre.next = head;return head;
}
var reverseLinkedList = function(head) {let cur = head;while(cur && cur.next) {cur = cur.next;}// 反转reverse(head);// 找到最后一个节点并作为最后的返回值return cur;
};
const newHead = reverseLinkedList(head);console.log(newHead);//> "head.val" 1
//> "head.val" 2
//> "head.val" 3
//> "head.val" 4
//> "pre.val" 5
//> "pre.val" 4
//> "pre.val" 3
//> "pre.val" 2
//> [object Object]

时间复杂度: O(n),空间复杂度: O(n)。


http://www.ppmy.cn/ops/50852.html

相关文章

材料科学SCI期刊,中科院3区,收稿范围广,易录用

一、期刊名称 International Journal of Material Forming 二、期刊简介概况 期刊类型:SCI 学科领域:材料科学 影响因子:2.4 中科院分区:3区 三、期刊征稿范围 该杂志发表和传播材料成型领域的原创研究。该研究应构成对材料…

2024-06-21力扣每日一题

链接: LCP 61. 气温变化趋势 题意 A、B两个数组,数组内相邻两个数字有大于、等于、小于三种变化情况,求最长的一段,使两个数组的这一段变化情况相同,并且不要求这一段只能有一种变化 解: 因为数组内只…

Gobject tutorial 九

The GObject messaging system Closures closure是一个抽象、通用的概念,它包含一个函数和一些变量。作用就是实现函数回调功能。 我们看看GLib对closure是怎么定义的。 /*** GClosure:* in_marshal: Indicates whether the closure is currently being invoked…

视频媒介VS文字媒介

看到一篇蛮有思考意义的文章就摘录下来了,也引起了反思 目录 一、视频的定义 二、”视频媒介“与”文字媒介”作对比 1.形象 VS 抽象 2.被动 VS 主动 三、视频的缺点-【更少】的思考 1.看视频为啥会导致【更少的思考】 2.内容的【浅薄化】 3.内容的【娱乐化…

算法设计与分析 笔记

截图摘自湖南大学彭鹏老师的ppt。笔记也是根据他的ppt整理的。 动态规划 核心 用数组记录中间结果,避免重复计算 三角数塔问题 问题描述 给定一个三角形数塔,从顶部出发,每次只能移动到下一行的相邻元素。要求找到一条路径,…

VMware安装及创建虚拟机

安装完成后,点击创建新的虚拟机 操作完成后就安装成功啦 ,下个教程出虚拟机Linux和xshell的连接及可能出现的问题解决方案

探索图神经网络(GNN):使用Python实现你的GNN模型

一、引言 图神经网络(Graph Neural Network, GNN)作为近年来机器学习和深度学习领域的热门话题,正逐渐吸引越来越多的研究者和开发者的关注。GNN能够处理图结构数据,在社交网络分析、推荐系统、化学分子结构预测等领域有着广泛的…

vscode 调试

VScode 调试教程 tasks.json和launch.json的设置(超详细)_vscode launch.json在哪-CSDN博客 launch.json {// Use IntelliSense to learn about possible attributes.// Hover to view descriptions of existing attributes.// For more information, v…