数据结构——Javascript中的链表(下)

前言

在上文中我们介绍了链表的概念以及如何在链表中执行增加,删除,查找的操作。以及链表与数组相比的优点与缺点。我们将在本文介绍如何使用链表解决LeetCode中的算法题。

正文

[ ] 我们先看第一题

js 复制代码

映入眼帘的是合并两个有序列表的标题,在知道链表的一些操作后面对这个题目先想想思路:我们要把6个结点串在一起,我们先将两个链表的第一个结点大小进行比较,穿上小的然后再将后面的结点与另一个链表上的结点进行比较,以此类推。最后剩余一个链表的结点不会进行比较,因为它是最大的结点

js 复制代码
function ListNode(val, next) {
    this.val = (val === undefined ? 0 : val)
    this.next = (next === undefined ? null : next)
}

var mergeTwoLists = function (list1, list2) {
    let head = new ListNode()
    let cur = head

    while (list1 && list2) {
        if (list1.val <= list2.val) {
            cur.next = list1
            list1 = list1.next
        } else {
            cur.next = list2
            list2 = list2.next  
        }
        cur = cur.next
    }
  

    cur.next = list1 !== null ? list1 : list2

    return head.next
};

在这里我们使用了 while 循环,在 list1list2 都非空时继续执行。循环内部比较两个链表当前节点的值: 如果 list1 当前节点的值小于等于 list2 当前节点的值,就会把 list1 的当前节点接到 cur 节点之后,并将 list1 指针移动到下一个节点。 否则,把 list2 的当前节点接到 cur 节点之后,并将 list2 指针移动到下一个节点。 当循环结束时,说明至少有一个链表已经遍历完。这时候会直接将没有遍历完的链表的剩余部分接到 cur.next,因为剩余部分已经是升序的,所以可以直接附加到结果链表末尾。

最后,返回新链表的头节点,即 head.next,因为在初始化时 head 节点的值是无意义的占位符,实际合并结果从 head.next 开始。

[ ] 题二

这道题是删除链表的结点,在上文中我们介绍过删除链表结点的操作。那么对于这道题,我们该如何去删除目标值所在的结点呢?

js 复制代码
val = 5

head = {
    val: 4,
    next: {
        val: 1,
        next: {
            val: 5,
            next: {
                val: 9,
                next: null
            }
        }
    }
}

假如这是给定的链表,这段代码给出val = 5,那么我们要让1的next直接指向9

js 复制代码
function ListNode(val) {
    this.val = val;
    this.next = null;
}
var deleteNode = function(head, val) {
    let cur = head 
    if(cur.val ===val){  // 要移除头结点
        head = head.next
    }

    while(cur && cur.next) {
        if(cur.next.val === val){
          // cur.next就是要被移除的节点
          cur.next = cur.next.next
             
        }


        cur = cur.next
    }

    return head;
};

因为我们要遍历这里链表,最后要返回头结点。所以我们先直接在最后面return head; 如果头节点就是要删除的节点,那么就直接将 head 指向下一个节点,这样相当于删除了原头节点

js 复制代码
      if(cur.next.val === val){
          // cur.next就是要被移除的节点
          cur.next = cur.next.next        
        }

然后排除了头结点就是目标结点的情况下,我们就进入循环遍历这个链表直到找到目标结点,然后就将 cur.next 指向 cur.next.next,从而跳过当前的 cur.next 节点。

[ ] 题三

这一题与上面一题相似度很高,在这里我们要去删除链表中重复的结点

js 复制代码
function ListNode(val, next) {
    this.val = (val === undefined ? 0 : val)
    this.next = (next === undefined ? null : next)
}
var deleteDuplicates = function (head) {
    if (!head || !head.next) return head; // 如果链表为空或只有一个元素,直接返回
    let cur = head;

    while (cur.next) {
        if (cur.val === cur.next.val) { // 发现重复元素
            cur.next = cur.next.next; // 删除当前节点的下一个节点
        } else {
            cur = cur.next; // 没有重复,移动到下一个节点
        }
    }

    return head;
};

在这段代码中我们先判断链表是否为空或者只有一个头结点,如果是的话就直接返回。不是的话就会进入循环当找到相同的元素时,就会 cur.next = cur.next.next;跳过这个重复的结点

总结

在上文学习了链表的相关知识后,这篇文章就是关于链表的实战,如果这些题目都没有问题的话,那你就基本掌握了链表这个知识点了。这篇文章对你有帮的话,那就留下你的点赞和收藏吧 :-)

相关推荐
Justinc.4 分钟前
CSS3新增边框属性(五)
前端·css·css3
武子康9 分钟前
大数据-212 数据挖掘 机器学习理论 - 无监督学习算法 KMeans 基本原理 簇内误差平方和
大数据·人工智能·学习·算法·机器学习·数据挖掘
fruge11 分钟前
纯css制作声波扩散动画、js+css3波纹催眠动画特效、【css3动画】圆波扩散效果、雷达光波效果完整代码
javascript·css·css3
neter.asia20 分钟前
vue中如何关闭eslint检测?
前端·javascript·vue.js
~甲壳虫20 分钟前
说说webpack中常见的Plugin?解决了什么问题?
前端·webpack·node.js
passer__jw76738 分钟前
【LeetCode】【算法】283. 移动零
数据结构·算法·leetcode
光影少年39 分钟前
vue2与vue3的全局通信插件,如何实现自定义的插件
前端·javascript·vue.js
As977_41 分钟前
前端学习Day12 CSS盒子的定位(相对定位篇“附练习”)
前端·css·学习
susu108301891143 分钟前
vue3 css的样式如果background没有,如何覆盖有background的样式
前端·css
Ocean☾44 分钟前
前端基础-html-注册界面
前端·算法·html