Loading [MathJax]/jax/output/CommonHTML/config.js
前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >面试链表不再怕

面试链表不再怕

作者头像
童欧巴
发布于 2020-07-16 15:57:24
发布于 2020-07-16 15:57:24
44000
代码可运行
举报
文章被收录于专栏:前端食堂前端食堂
运行总次数:0
代码可运行

这是前端食堂的第35篇原创

「口味:蒜蓉荷兰豆」

「烹饪时间:8min」

本文已收录在前端食堂同名仓库Github github.com/Geekhyt,欢迎光临食堂,如果觉得酒菜还算可口,赏个 Star 对食堂老板来说是莫大的鼓励。

链表

数组想必大家都很熟悉,几乎我们每天都会操作它。那么我们就来对比数组来学习链表,首先要明确的是,链表和数组的底层存储结构不同,数组要求存储在一块连续的内存中,而链表是通过指针将一组零散的内存块串联起来。可见链表对内存的要求降低了,但是随机访问的性能就没有数组好了,需要 O(n) 的时间复杂度。

下图中展示了单链表及单链表的添加和删除操作,其实链表操作的本质就是处理链表结点之间的指针。

在删除链表结点的操作中,我们只需要将需要删除结点的前驱结点的 next 指针,指向其后继即可。这样,当前被删除的结点就被丢弃在内存中,等待着它的是被垃圾回收器清除。

为了更便于你理解,链表可以类比现实生活中的火车,火车的每节车厢就是链表的一个个结点。车厢之间相互连接,可以添加或者移除掉。春运时,客运量比较大,列车一般会加挂车厢。

链表的结点结构由数据域和指针域组成,在 JavaScript 中,以嵌套的对象形式实现。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
{
    // 数据域
    val: 1,
    // 指针域
    next: {
        val:2,
        next: ...
    }
}  

名词科普

  • 头结点:头结点用来记录链表的基地址,是我们遍历链表的起点
  • 尾结点:尾结点的指针不是指向下一个结点,而是指向一个空地址 NULL
  • 单链表:单链表是单向的,它的结点只有一个后继指针 next 指向后面的结点,尾结点指针指向空地址
  • 循环链表:循环链表的尾结点指针指向链表的头结点
  • 双向链表:双向链表支持两个方向,每个结点不止有一个后继指针 next 指向后面的结点,还有一个前驱指针 prev 指向前面的结点,双向链表会占用更多的内存,但是查找前驱节点的时间复杂度是 O(1) ,比单链表的插入和删除操作都更高效
  • 双向循环链表

循环链表

双向链表

双向循环链表

LeetCode真题

掌握了链表的基础知识后,我们拿几道链表的 LeetCode 真题练练手。

1.合并两个有序链表

https://leetcode-cn.com/problems/merge-two-sorted-lists/
思路
  • 使用递归来解题
  • 将两个链表头部较小的一个与剩下的元素合并
  • 当两条链表中的一条为空时终止递归
复杂度分析

N+M 是两条链表的长度

  • 时间复杂度:O(M+N)
  • 空间复杂度:O(M+N)
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
const mergeTwoLists = function (l1, l2) {
    if (l1 === null) {
        return l2;
    }
    if (l2 === null) {
        return l1;
    }
    if (l1.val < l2.val) {
        l1.next = mergeTwoLists(l1.next, l2);
        return l1;
    } else {
        l2.next = mergeTwoLists(l1, l2.next);
        return l2;
    }
};

2.环形链表

https://leetcode-cn.com/problems/linked-list-cycle/
思路
  • 双指针法
  • 使用快慢不同的两个指针遍历,快指针一次走两步,慢指针一次走一步
  • 如果没有环,快指针会先到达尾部,返回 false
  • 如果有环,则一定会相遇
复杂度分析
  • 时间复杂度:O(N)
  • 空间复杂度:O(1)
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
const hasCycle = function(head) {
    if (!head || !head.next) {
        return false;
    }
    let fast = head.next;
    let slow = head;
    while (fast !== slow) {
        if (!fast || !fast.next) {
            return false;
        }
        fast = fast.next.next;
        slow = slow.next;
    }
    return true;
};
思路
  • 标记法
  • 遍历链表,通过标记判断是否有环,如果标记存在则有环。
复杂度分析
  • 时间复杂度:O(N)
  • 空间复杂度:O(1)
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
const hasCycle = function(head) {
    while (head) {
        if (head.flag) {
            return true;
        } else {
            head.flag = true;
            head = head.next;
        }
    }
    return false;
}

3.反转链表

https://leetcode-cn.com/problems/reverse-linked-list/
思路
  • 迭代
  • 初始化前驱节点为 null,初始化目标节点为头节点
  • 遍历链表,记录 next 节点并反转指针
  • prev 和 curr 指针分别往前移动一步
  • 反转结束后,prev 成为新链表的头节点
复杂度分析
  • 时间复杂度:O(N)
  • 空间复杂度:O(1)
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
const reverseList = function(head) {
    let prev = null;
    let curr = head;
    while (curr !== null) {
        let next = curr.next;
        curr.next = prev;
        prev = curr;
        curr = next;
    }
    return prev;
};

4.删除结点的倒数第 N 个节点

https://leetcode-cn.com/problems/remove-nth-node-from-end-of-list/
思路
  • 删除倒数第 n 个结点,我们需要找到倒数第 n+1 个结点,删除其后继结点即可
  • 添加 prev 结点,也称其为哨兵结点,处理边界问题
  • 使用双指针法,快指针先走 n+1 步,然后快慢指针同步往前走,直到 fast.next 为 null
  • 删除倒数第 n 个结点,返回 prev.next
复杂度分析
  • 时间复杂度:O(N)
  • 空间复杂度:O(1)
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
const removeNthFromEnd = function(head, n) {
    let prev = new ListNode(0);
    prev.next = head;
    let fast = prev;
    let slow = prev;
    while (n--) {
        fast = fast.next;
    }
    while (fast && fast.next) {
        fast = fast.next;
        slow = slow.next;
    }
    slow.next = slow.next.next;
    return prev.next;
};

5.求链表的中间结点

https://leetcode-cn.com/problems/middle-of-the-linked-list/
思路
  • 双指针法
  • 使用快慢不同的两个指针遍历,快指针一次走两步,慢指针一次走一步
  • 当快指针到达终点时,慢指针刚好走到中间
复杂度分析
  • 时间复杂度:O(N) N 是给定链表的结点数目
  • 空间复杂度:O(1) 只需要常数空间存放 slow 和 fast 两个指针
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
const middleNode = function(head) {
    let fast = head;
    let slow = head;
    while (fast && fast.next) {
        slow = slow.next;
        fast = fast.next.next;
    }
    return slow;
本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2020-07-12,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 前端食堂 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
链表+6道前端算法面试高频题解
数组在上一篇的专栏,数组回炉重造+6道前端算法面试高频题解中我们进行了回顾和刷题。
童欧巴
2021/04/09
3360
链表+6道前端算法面试高频题解
面试官系列 - LeetCode链表知识点&题型总结
前段时间,写了面试必备的一系列文章,反应还不错。有一些读者反馈说,能不能整理一些面试常见的算法。前段时间,我恰好总结了 LeetCode 常见的面试算法题目。今天给大家分享一下。
程序员徐公
2022/01/20
7160
面试官系列 - LeetCode链表知识点&题型总结
LeetCode链表知识点&题型总结
刚开始准备刷算法题目的时候,感觉真的是好难,十道题目有九道是不会的。心中曾一万只草泥马跑过,自己怎么这么辣鸡。
程序员徐公
2020/08/04
1.7K0
LeetCode链表知识点&题型总结
LeetCode通关:听说链表是门槛,这就抬脚跨门而入
一个单向链表包含两个值: 当前节点的值和一个指向下一个节点的引用。也可以称之为数据域和指针域。
三分恶
2021/07/27
4510
2023年前端面试题汇总-数据结构(链表)
在计算机里,不保存在连续存储空间中,而每一个元素里都保存了到下一个元素的地址的数据结构,我们称之为链表(Linked List)。链表上的每一个元素又可以称它为节点(Node),而链表中第一个元素,称它为头节点(Head Node),最后一个元素称它为尾节点(Tail Node)。
越陌度阡
2023/06/10
1.1K0
2023年前端面试题汇总-数据结构(链表)
Leetcode链表题目
为了表示给定链表中的环,我们使用整数 pos 来表示链表尾连接到链表中的位置(索引从 0 开始)。如果 pos 是 -1,则在该链表中没有环。
润森
2019/11/27
4020
Leetcode链表题目
LeetCode笔记 | 链表(ing)
思路如下: 0.利用递归首先找到单链表的最后一个节点; 最后一个节点存储在re里面, re在找到最后一个节点时被赋值且其永远为最后一个节点的值,保持不变; 从找到最后一个节点开始, 从最后往前的方向,每一层递归反转一对节点 / 一个指向;
凌川江雪
2019/10/24
4820
LeetCode笔记 | 链表(ing)
206. 反转链表 & 876. 链表的中间结点
翻转链表也是很经典的链表相关题目,属于必须掌握的。需要声明两个指针,分别指向前驱节点和当前节点,而翻转链表的核心就是当前节点的next指向的变化。
chuckQu
2022/08/19
3340
算法细节系列(31):链表
版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/u014688145/article/details/72902779
用户1147447
2019/05/26
4250
36 张图带你深刻理解链表
在浅谈数组这篇文章中,我们说到数组在内存中是用一块连续的内存空间存储的。由于是用连续内存空间存储的,那么就会出现,明明还剩50M内存,但由于不是连续的,而导致在创建一个大小为50M的数组时,申请内存失败。
五分钟学算法
2021/03/10
8360
马上2021年了线性表你还不知道原理?给老王整的明明白白
线性表是n个数据元素的有限序列,最常用的是链式表达,通常也叫做线性链表或者链表在链表中存储的数据元素也叫做结点,一个结点存储的就是一条数据记录。
手撕代码八百里
2020/10/26
3840
七十一、去重交换排序链表、 求链表的中间结点
最近在重新梳理学算法的知识,本文为链表常见操作复习的总结文章,会讲解常见的链表题目实现思路及附上答案,这些题目在leetcode上对应的题号也有给出,好好学习算法吧~
润森
2022/08/17
4550
七十一、去重交换排序链表、 求链表的中间结点
《剑指 Offer (第 2 版)》链表部分 JavaScript 题解
《剑指 Offer(第 2 版)》通行全球的程序员经典面试秘籍。剖析典型的编程面试题,系统整理基础知识、代码质量、解题思路、优化效率和综合能力这 5 个面试要点。
用户8921923
2022/10/24
2570
《剑指 Offer (第 2 版)》链表部分 JavaScript 题解
七十、反转和合并链表、 链表有环的判断
最近在重新梳理学算法的知识,本文为链表常见操作复习的总结文章,会讲解常见的链表题目实现思路及附上答案,这些题目在leetcode上对应的题号也有给出,好好学习算法吧~
润森
2022/08/17
4840
七十、反转和合并链表、 链表有环的判断
搞定大厂算法面试之leetcode精讲7.双指针
方法2.c=-(a+b): 确定了a和b,那就可以想两数之和一样,在map中寻找-(a+b),减少一层循环,时间复杂度O(n^2),空间复杂度O(n)。
全栈潇晨
2021/11/26
3350
LeetCode,判断一个链表是否为回文链表
空间复杂度:O(n),这种解法,我们使用了一个数组列表存放链表的元素值。n 指的是链表的元素个数。
微客鸟窝
2021/08/18
3550
LeetCode,判断一个链表是否为回文链表
JS算法探险之链表
说起,「链表」在前端领域,可能有些许的陌生。但是,它作为一个传统的数据结构,在一些高阶领域还是有用武之地的。
前端柒八九
2022/08/25
5560
JS算法探险之链表
剑指offer | 面试题17:链表中倒数第k个节点
题目描述: 输入一个链表,输出该链表中倒数第k个节点。为了符合大多数人的习惯,本题从1开始计数,即链表的尾节点是倒数第1个节点。
千羽
2021/12/29
2220
剑指offer | 面试题17:链表中倒数第k个节点
数组与链表
数组(Array)是一种线性表数据结构。它用一组连续的内存空间,来存储一组具有相同类型的数据。
Yif
2019/12/26
6160
LeetCode题解—链表中环的检测
如果链表中有某个节点,可以通过连续跟踪 next 指针再次到达,则链表中存在环。 pos为环的起点位置。
码上积木
2021/03/10
1.3K0
相关推荐
链表+6道前端算法面试高频题解
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验