算法 day4 【双指针、快慢指针、环形链表】链表下

news2024/9/17 7:37:02

⚡刷题计划day4继续,可以点个免费的赞哦~

下一期将会开启哈希表刷题专题,往期可看专栏,关注不迷路,

您的支持是我的最大动力🌹~

目录

⚡刷题计划day4继续,可以点个免费的赞哦~

下一期将会开启哈希表刷题专题,往期可看专栏,关注不迷路,

您的支持是我的最大动力🌹~

题目一:19. 删除链表的倒数第 N 个结点

法一:计算链表长度

法二:双指针

题目二:面试题 02.07. 链表相交

法一:双指针

法二:合并链表实现同步移动

题目三:142. 环形链表 II

法一:快慢指针法

1.判断链表是否有环

2.如果有环,如何找到环的入口

2.1 n==1

2.2 n>1

法二:哈希表


题目一:19. 删除链表的倒数第 N 个结点

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

(https://leetcode.cn/problems/remove-nth-node-from-end-of-list/description/)

法一:计算链表长度

常规思路:先遍历得到链表长度L,然后再次遍历到 L-n+1个结点,便是我们需要删除的结点;

这里我们还是使用虚拟头结点统一,于是从虚拟结点开始遍历L-n+1个结点,它的下一个结点便是我们要删除的结点,这样我们只需修改一次指针,就可完成删除操作。

如图辅助理解:

AC代码

class Solution {
    public ListNode removeNthFromEnd(ListNode head, int n) {
        ListNode dummyHead = new ListNode();
        dummyHead.next=head;
        ListNode cur = dummyHead;
        int length = getLength(head);
        for (int i=1;i<length-n+1;i++){
            cur = cur.next;
        }
        if(cur.next!=null){//避免空指针
            cur.next = cur.next.next;//删除操作
        }
        
        return dummyHead.next;
​
    }
    public int getLength(ListNode head){
        int length = 0;
        while(head != null){
            length++;
            head=head.next;
        }
        return length;
    }
}

法二:双指针

主要思路:

要删除倒数第n个,我们可以使用双指针,这样不用去求长度;

保持一个相差n的区间,fast在前,slow在后。这样等fast走到链表末尾,slow对应的就是我们要删除的结点;

因为链表删除我们需要通过前一个结点,所以将相差区间设为n+1,可以结合图理解:

AC代码

class Solution {
    public ListNode removeNthFromEnd(ListNode head, int n) {
        ListNode dummyHead = new ListNode();
        dummyHead.next=head;
        ListNode fast = dummyHead;
        ListNode slow = dummyHead;
​
        // 只要快慢指针相差 n 个结点即可
        for (int i=1;i<=n+1;i++){
            fast = fast.next;
        }
        while (fast!=null){
            fast = fast.next;
            slow = slow.next;
        }
​
        if(slow.next!=null){
            slow.next = slow.next.next;
        }
        return dummyHead.next;
    }
}

题目二:面试题 02.07. 链表相交

leetcode:面试题 02.07. 链表相交

(https://leetcode.cn/problems/intersection-of-two-linked-lists-lcci/description/)

简单来说,就是求两个链表交点节点的指针。 这里要注意,交点不是数值相等,而是指针相等。

为了方便举例,假设节点元素数值相等,则节点指针相等。

看如下两个链表,目前curA指向链表A的头结点,curB指向链表B的头结点:

我们求出两个链表的长度,并求出两个链表长度的差值,然后让curA移动到,和curB 末尾对齐的位置,如图

此时我们就可以比较curA和curB是否相同,如果不相同,同时向后移动curA和curB,如果遇到curA == curB,则找到交点。

否则循环退出返回空指针。

AC代码

法一:双指针

public class Solution {
​
    public ListNode getIntersectionNode(ListNode headA, ListNode headB) {
        // 初始化两个链表的当前节点
        ListNode cur1 = headA;
        ListNode cur2 = headB;
        // 初始化两个链表的长度
        int len1 = 0;
        int len2 = 0;
​
        // 求cur1长度
        while (cur1 != null) {
            len1++;
            cur1 = cur1.next;
        }
        // 求cur2长度
        while (cur2 != null) {
            len2++;
            cur2 = cur2.next;
        }
​
        // 重新初始化两个链表的当前节点
        cur1 = headA;
        cur2 = headB;
​
        // 如果第一个链表比第二个短,交换两个链表的头节点和长度
        if (len1 < len2) {
            //1. swap (len1, len2);
            int temp = len1;
            len1 = len2;
            len2 = temp;
            //2. swap (cur1, cur2);
            ListNode temNode = cur1;
            cur1 = cur2;
            cur2 = temNode;
        }
​
        // 计算长度的差值
        int gap = len1 - len2;
        // 移动较长链表的当前节点,使cur1与cur2的末尾位置对齐,看图
        while (gap-- > 0) {
            cur1 = cur1.next;
        }
​
        // 同时遍历两个链表,遇到相同则直接返回
        while (cur1 != null) {
            if (cur1 == cur2) {
                return cur1; // 返回相交的节点
            }
            cur1 = cur1.next;
            cur2 = cur2.next;
        }
​
        // 如果两个链表没有相交,返回null
        return null;
    }
}

法二:合并链表实现同步移动

主要思路:

  1. 同步移动指针

    • 使用一个 while 循环,只要 p1p2 没有相遇,就继续移动指针。

    • 对于 p1,如果它到达了链表 A 的末尾(即 p1null),则将它移动到链表 B 的头部,重新开始遍历。

    • 对于 p2,如果它到达了链表 B 的末尾(即 p2null),则将它移动到链表 A 的头部,重新开始遍历。

  2. 相遇即相交

    • 当两个指针 p1p2 相遇时,它们指向的就是链表相交的节点。因为两个指针最终都会遍历完两个链表,如果链表有相交点,它们最终会在相交点相遇。

  3. 返回结果

    • 一旦 p1p2 相遇,即它们指向同一个节点,就返回这个节点。

 public class Solution {
    public ListNode getIntersectionNode(ListNode headA, ListNode headB) {
        // p1 指向 A 链表头结点,p2 指向 B 链表头结点
        ListNode p1 = headA, p2 = headB;
        while (p1 != p2) {
            // p1 走一步,如果走到 A 链表末尾,转到 B 链表
            if (p1 == null) p1 = headB;
            else            p1 = p1.next;
            // p2 走一步,如果走到 B 链表末尾,转到 A 链表
            if (p2 == null) p2 = headA;
            else            p2 = p2.next;
        }
        return p1;
    }
}

题目三:142. 环形链表 II

leetcode:142. 环形链表 II

(https://leetcode.cn/problems/linked-list-cycle-ii/description/)

法一:快慢指针法

判断链表是否有环,我们一般可以使用快慢指针法

此题还是有一定难度,考察了对链表环的判断,已经需要进行数学运算,下文会进行详细解释。

对于此题大致分为两大步:

1.判断链表是否有环

2.如果有环,如何找到环的入口


1.判断链表是否有环

分别定义fast,slow指针,fast指针每次移动两个节点,slow指针每次移动一个节点,如果 fast 和 slow指针在途中相遇 ,说明这个链表有环。

2.如果有环,如何找到环的入口

这步已经可以判断链表是否有环了,接下来是寻找环的入口,需要用到一点数学运算。

假设从头结点到环形入口节点 的节点数为x。 环形入口节点到 fast指针与slow指针相遇节点 节点数为y。 从相遇节点 再到环形入口节点节点数为 z。 如图所示:

那么当相遇时:

slow指针走过的节点数为: x + y, fast指针走过的节点数:x + y + n (y + z),n为fast指针在环内走了n圈才遇到slow指针, (y+z)为 一圈内节点的个数A。

加深理解注:

1.fast指针为什么 n (y + z)? 因为fast比slow快,fast进入圈后至少需要一圈后追反超slow,由此也可知道,n>=1(后续解题会用)。

2.为什么slow就不用比如k(y+z)? 因为slow进入圈后在一圈内一定会被fast追上

3.那为什么slow一圈内就一定会被fast追上呢? 可以这样通俗理解:首先fast会比slow快1,如果slow走一圈,fast就会走两圈,那一定会追上。

因为fast指针是一步走两个节点,slow指针一步走一个节点, 所以 slow指针走过的节点数 * 2 = fast指针走过的节点数,所以可以列出方程:

(x + y) * 2 = x + y + n (y + z)

因为我们要找环形入口,即需要找x,将x单独放出来化简:

x = n (y + z) - y

但是我们看一下这个式子呢,也发现不了什么,我们是想通过右边的参数来求x,但发现目前右侧也没啥特殊形式,还有个-y。于是我们可以提一个(y+z)出来,

x = (n - 1) (y + z) + z(此处n>=1,前面有解释)

此时就明了了。


2.1 n==1

当 n为1的时候,公式就化解为 x = z

这就意味着,从头结点出发一个指针,从相遇节点也出发一个指针,这两个指针每次只走一个节点, 那么当这两个指针相遇的时候就是 环形入口的节点

也就是在相遇节点处,定义一个指针index1,在头结点处定一个指针index2。

让index1和index2同时移动,每次移动一个节点, 那么他们相遇的地方就是 环形入口的节点。

2.2 n>1

那么 n如果大于1是什么情况呢,就是fast指针在环形转n圈之后才遇到 slow指针。

其实这种情况和n为1的时候效果是一样的,一样可以通过这个方法找到 环形的入口节点,只不过,index1 指针在环里 多转了(n-1)圈,然后再遇到index2,相遇点依然是环形的入口节点。

代码如下:

public class Solution {
    public ListNode detectCycle(ListNode head) {
        ListNode fast = head;
        ListNode slow = head;
        while(fast!=null && fast.next!=null){
            fast = fast.next.next;
            slow = slow.next;
​
            //相遇,有环
            if(fast==slow){
                ListNode index1  = fast;
                ListNode index2  = head;
                // 两个指针,从头结点和相遇结点,各走一步,直到相遇,相遇点即为环入口
                while (index1 != index2){
                    index1 = index1.next;
                    index2 = index2.next;
                }
                return index1;
            }
​
        }
        return null;
    }
}

法二:哈希表

这个思路就简单很多,时间复杂度:O(N),空间复杂度:O(N);

但第一种双指针的解法也需要掌握,时间复杂度:O(N),空间复杂度:O(1)。

关于哈希表我们下期也会做相应的专题刷题。

主要思路:遍历链表中的每个节点,并将它记录下来;一旦遇到了此前遍历过的节点,就可以判定链表中存在环。借助哈希表可以很方便地实现。

代码:

详细见注解

public class Solution {
    public ListNode detectCycle(ListNode head) {
        // 初始化指针pos指向头结点
        ListNode pos = head;
        // 使用HashSet来存储已经访问过的节点
        Set<ListNode> visited = new HashSet<ListNode>();
​
        // 遍历链表
        while (pos != null) {
            // 如果当前节点已经被访问过,说明存在环,返回当前节点
            if (visited.contains(pos)) {
                return pos;
            } else {
                // 否则,将当前节点添加到visited集合中
                visited.add(pos);
            }
            // 移动到下一个节点
            pos = pos.next;
        }
​
        // 如果遍历完整个链表都没有发现环,返回null
        return null;
    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1947789.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

无法连接网络打印机0x00000709原因分析及多种解决方法

在日常办公和生活中&#xff0c;打印机是不可或缺的重要设备。然而&#xff0c;有时在连接打印机的过程中&#xff0c;我们可能会遇到错误代码0x00000709的提示。有更新补丁导致的、有访问共享打印机服务异常、有访问共享打印机驱动异常等问题导致的&#xff0c;针对访问共享打…

实验三 FPGA使用Verilog HDL设计加法器

实验目的 掌握使用Vivado软件进行设计、综合、仿真、布线的方法。掌握FPGA程序的下载方法。掌握使用Verilog HDL设计加法器的方法。 实验要求 采用Verilog HDL语言设计加法器&#xff0c;实现两个4位数的相加运算&#xff0c;并将结果通过LED灯或数码管显示出来。对设计进行综…

如何通过集成软件授权管理系统推动企业业务增长?

软件货币化已经成为许多企业商业成功的关键&#xff0c;随着全球数字化进程不断深入&#xff0c;其重要性也在不断增加。将许可解决方案优化集成到现有系统中&#xff0c;已成为从接收到订单到交付和激活许可的任何高效流程的基本要素。 软件货币化无处不在 无论是传统的软件企…

[言简意赅] Matlab生成FPGA端rom初始化文件.coe

&#x1f38e;Matlab生成FPGA端rom初始化文件.coe 本文主打言简意赅。 函数源码 function gencoeInitialROM(width, depth, signal, filepath)% gencoeInitialROM - 生成 Xilinx ROM 初始化格式的 COE 文件%% 输入参数:% width - ROM 数据位宽% depth - ROM 数据深度% s…

在 LCD 上显示 png 图片-I.MX6U嵌入式Linux C应用编程学习笔记基于正点原子阿尔法开发板

在 LCD 上显示 png 图片 PNG 简介 无损压缩&#xff1a;PNG 使用 LZ77 派生算法进行无损压缩&#xff0c;确保图像质量不受损&#xff0c;且压缩比高 体积小&#xff1a;通过高压缩比&#xff0c;PNG 文件体积小&#xff0c;适合网络传输 索引彩色模式&#xff1a;PNG-8 格式…

Unity UGUI 之 RectTransform

本文仅作学习笔记与交流&#xff0c;不作任何商业用途 本文包括但不限于unity官方手册&#xff0c;唐老狮&#xff0c;麦扣教程知识&#xff0c;引用会标记&#xff0c;如有不足还请斧正 Unity - Manual: Rect Transform 1.Rect Transform是什么 2.轴心与锚点的映射关系 首先…

获取后端返回的图形验证码

如果后端返回的直接就是一个图形&#xff0c;有以下几种方式展示 一、直接在img标签里面的src里面调用接口 <img :src"dialogSrc" class"photo" alt"验证码图片" click"changeDialog">let orgUrl "/api/captcha" …

论文解读:DiAD之SG网络

目录 一、SG网络功能介绍二、SG网络代码实现 一、SG网络功能介绍 DiAD论文最主要的创新点就是使用SG网络解决多类别异常检测中的语义信息丢失问题&#xff0c;那么它是怎么实现的保留原始图像语义信息的同时重建异常区域&#xff1f; 与稳定扩散去噪网络的连接&#xff1a; S…

机器学习(二十):偏差和方差问题

一、判断偏差和方差 以多项式回归为例&#xff0c;红点为训练集数据&#xff0c;绿点为交叉验证数据。 下图的模型&#xff0c;训练集误差大&#xff0c;交叉验证集误差大&#xff0c;这代表偏差很大 下图的模型&#xff0c;训练集误差小&#xff0c;交叉验证集误差小&#x…

Linux网络:传输层协议TCP(二)三次挥手四次握手详解

目录 一、TCP的连接管理机制 1.1三次握手 1.2四次挥手 二、理解 TIME_WAIT 状态 2.1解决TIME_WAIT 状态引起的 bind 失败的方法 三、理解CLOSE_WAIT状态 一、TCP的连接管理机制 在正常情况下, TCP 要经过三次握手建立连接, 四次挥手断开连接 1.1三次握手 三次握手顾名思…

vue import from

vue import from 导入文件&#xff0c;从XXXX路径&#xff1b;引入文件 import xxxx from “./minins/resize” import xxxx from “./minins/resize.js” vue.config.js 定义 : resolve(src)&#xff1b;就是指src 目录 import xxxx from “/utils/auth” im…

vue3知识

目录 基础vue开发前的准备vue项目目录结构模板语法属性绑定条件渲染列表渲染通过key管理状态事件处理事件传参事件修饰符数组变化侦测计算属性Class绑定style绑定侦听器表单输入绑定模板引用组件组成组件嵌套关系组件注册方式组件传递数据Props(父传子)组件传递多种数据类型组件…

怎么批量加密U盘?U盘批量加密的方法有哪些?

加密U盘是保护U盘数据安全的重要方法。而当需要加密的U盘数量较多时&#xff0c;我们需要批量加密U盘。那么&#xff0c;U盘怎么批量加密呢&#xff1f;下面我们就来了解一下。 U盘内存卡批量只读加密专家 U盘内存卡批量只读加密专家是一款专业的U盘加密软件&#xff0c;适用于…

什么牌子的充电宝又好又耐用?认准这几个充电宝品牌!错过就吃亏

在 2024 年&#xff0c;充电宝已然成为我们生活中不可或缺的电子配件。但面对市场上琳琅满目的充电宝产品&#xff0c;如何挑选出一款适合自己的&#xff0c;却让许多人感到困惑。充电宝要怎么挑&#xff1f;这可不是一个简单的问题。不同的使用场景、不同的设备需求&#xff0…

02 MySQL数据库管理

目录 1.数据库的结构 sql语言主要由以下几部分组成 2. 数据库与表的创建和管理 1&#xff0c;创建数据库 2&#xff0c;创建表并添加数据 3&#xff0c;添加一条数据 4&#xff0c;查询数据 5&#xff0c;更新数据 6&#xff0c;删除数据 3.用户权限管理 1.创建用户 …

3万多有分类的成语词典ACCESS\EXCEL数据库

今天最后发一个成语词典的数据库了&#xff0c;因为成语词典的数据库太多了导致我自己都有些糊涂了&#xff0c;今天这份数据库应该说是最好的成语词典了&#xff0c;不但包含了3级分类&#xff0c;而且还有级别&#xff08;不要太较真&#xff09;字段。 数据库包含多个表&…

利用 Databend 生态构建现代数据湖工作流

数据是洞察力的基石&#xff0c;越来越多的企业开始建设以数据资产为中心的存储和分析一体化方案&#xff0c;这要求 Data Infra 架构能够提供可扩展、灵活且统一的数据工作流。现代数据湖架构同时兼顾数据湖的可扩展性和数据仓库的性能&#xff0c;满足对大规模数据处理的需求…

视频文件怎么压缩到最小 视频文件怎么压缩到最小内存 4个简单的方法工具分享简单步骤

如何压缩大视频文件以减小其大小&#xff1f;在分享或存储大视频文件时&#xff0c;有效压缩是关键&#xff0c;以降低文件大小且不显著牺牲视觉和听觉质量。视频文件的大小直接影响传输、分享和存储的成本与便捷性。掌握压缩视频的技能对于数字内容处理至关重要&#xff0c;能…

【Android】linux

android系统就是跑在linux上的系统。Linux层里面包含系统和硬件驱动等一些本地代码的环境。 linux的目录 mount: 用于查看哪个模块输入只读&#xff0c;一般显示为&#xff1a; [rootlocalhost ~]# mount /dev/cciss/c0d0p2 on / type ext3 (rw) proc on /proc type proc (…

SpringBoot 实现图形验证码

一、最终结果展示 二、前端代码 2.1 index.html <!DOCTYPE html> <html lang"en"><head><meta charset"utf-8"><title>验证码</title><style>#inputCaptcha {height: 30px;vertical-align: middle;}#verifica…