IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 数据结构与算法 -> 跟着代码随想录练算法——链表(js) -> 正文阅读

[数据结构与算法]跟着代码随想录练算法——链表(js)

203. 移除链表元素

考虑到可能要移除链表头结点,可以直接在链表上操作,但是这里需要作为一种情况单独考虑;也可以在链表头结点之前再添加一个节点,这样移除头结点也可以有像移除其他节点一样的操作,不需要特判

直接在链表上操作

var removeElements = function(head, val) {
    
    while(head && head.val==val) head=head.next
    if(!head) return null
    let cur = head.next
    let pre = head
    while(cur){
        if(cur.val==val){
            pre.next=cur.next
        }else pre = cur
        cur=cur.next
    }
    return head
};

添加一个新的结点

function ListNode(val, next) {
     this.val = (val===undefined ? 0 : val)
     this.next = (next===undefined ? null : next)
 }
var removeElements = function(head, val) {
    let node = new ListNode(-1, head)
    let cur = node
    while(node.next){
        if(node.next.val === val){
            node.next = node.next.next
        }else node = node.next
        
    }
    return cur.next
};

206. 反转链表

var reverseList = function(head) {
    let pre = null
    let cur = head
    let next
    while(cur){
        next = cur.next
        cur.next = pre
        pre = cur
        cur = next
    }
    return pre
};

24. 两两交换链表中的节点

  • new 一个虚拟的头结点 node 指向 head,p1p2 为当前需要交换的两个节点,next 作为除去当前交换节点的第一个节点
  • while 循环遍历链表,循环结束的条件是 p1 为空或者 p2 为空
  • p1 为空对应偶数个节点的链表操作结束,p2 为空对应奇数个节点的链表操作结束
  • while循环体:
    • 先将后续链表节点用next 存下来
    • pre 的下一个指向 p2
    • p2 的下一个指向 p1
    • p1 的下一个指向 next
    • 到这里两个节点的交换就完成了 ,从 pre -> p1 -> p2 -> next 到 pre -> p2 -> p1 -> next,接下来就是后移节点继续定位到下一对需要交换的节点了
    • pre 指向 p1
    • p1 指向 next,若发现p1 为空,则证明偶数个节点的链表已经交换结束,break 出去
    • p2 指向next 的next,若发现p1 为空,则证明奇数个节点的链表已经交换结束
  • 最终返回 node.next
function ListNode(val, next) {
  this.val = (val===undefined ? 0 : val)
  this.next = (next===undefined ? null : next)
}
var swapPairs = function(head) {
    if(!head || !head.next) return head
    let node = new ListNode(-1,head)
    let pre = node
    let p1 = node.next
    let p2 = p1.next
    let next = null
    while(p2){
        next = p2.next
        pre.next = p2
        p2.next = p1
        p1.next = next
        pre = p1
        p1 = next
        if(p1 === null) break 
        p2 = next.next
    }
    return node.next
};

19. 删除链表的倒数第 N 个结点

这题可以先找到倒数第 n-1个节点,然后直接删掉它的下一个节点。因为有可能要删掉头结点,所以给这个链表添加了一个虚拟头结点 node。如何遍历一遍找到倒数第 n-1 个节点呢?使用双指针 p1,p2,p1初始化指向 node,p2初始化指向 node.next,p2往后走一步,n 就 --,当 n <= 0的时候 ,p1也往后移动,当 p2 指向null的时候,p1指向倒数第n-1个节点。

function ListNode(val, next) {
  this.val = (val===undefined ? 0 : val)
  this.next = (next===undefined ? null : next)
}
var removeNthFromEnd = function(head, n) {
    if(!head) return head
    let node = new ListNode(-1,head)
    let p1 = node
    let p2 = node.next  // 快指针
    while(p2){
        p2 = p2.next
        if(n <= 0) p1 = p1.next
        n --
    }
    console.log(p1.val)
    // 结束循环之后 p1 指向需要删除节点的前一个
    p1.next = p1.next.next
    return node.next
};

面试题 02.07. 链表相交

两个链表长度不一样如何确定交点起始位置呢?可以先让p1把headA走一遍,p2把headB走一遍,当p1走完时,再去走p2,当p2走完时再去走p1,第二趟如果两者相等,则表白此时出去公共部分的起始位置

  • flagA,flagB 表示两个指针是否已经换过链表遍历了
  • 循环的结束条件为 p1,p1都不为空,如果有一个为空都表示没有公共部分
  • 当两个指针都已经换过链表遍历并且当前相等,则返回当前节点
  • 当指针走到链表末尾,判断flagA,flagB是否为0,为0表示还没有换过链表,那就换另外一个链表遍历,不为0表示之前已经更换过了,直接跳出循环
var getIntersectionNode = function(headA, headB) {
    let flagA = 0
    let flagB = 0
    let p1 = headA
    let p2 = headB
    while(p1&&p2){
        if(flagA && flagB && p1 === p2){
            return p1
        }
        if(p1.next){
            p1 = p1.next
        } 
        else{
            if(flagA == 0){
                flagA = 1
                p1 = headB
            }else break
        }
        if(p2.next){
            p2 = p2.next
        }else{
            if(flagB == 0){
                p2 = headA
                flagB = 1
            }else break
        }
    }
    return null
};

142. 环形链表 II

第二遍做这个题了,还是没有根据快慢指针的路程得出那个需要的等式,又看了一遍题解的推导过程,希望我第三遍遇到这个题我能自己写出来吧,呜呜呜。

var detectCycle = function(head) {
    if(!head || !head.next) return null
    let slow = head
    let fast = head
    let p
    while(slow && fast){
        //slow fast 只有有一个走到空 就说明没有环
        if(slow.next) slow = slow.next
        else return null
        if(fast.next && fast.next.next) fast = fast.next.next
        else return null
        // 第一次相遇
        if(slow === fast){
            p = head
            break
        }
    }
    while(p && slow){
        if(p === slow) return p
        if(p && p.next) p = p.next
        if(slow.next) slow = slow.next
    }
    return null
};
  数据结构与算法 最新文章
【力扣106】 从中序与后续遍历序列构造二叉
leetcode 322 零钱兑换
哈希的应用:海量数据处理
动态规划|最短Hamilton路径
华为机试_HJ41 称砝码【中等】【menset】【
【C与数据结构】——寒假提高每日练习Day1
基础算法——堆排序
2023王道数据结构线性表--单链表课后习题部
LeetCode 之 反转链表的一部分
【题解】lintcode必刷50题<有效的括号序列
上一篇文章      下一篇文章      查看所有文章
加:2022-05-18 17:54:13  更:2022-05-18 17:55:17 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/9 2:00:42-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码