【数据结构与算法 | 灵神题单 | 快慢指针(链表)篇】力扣876, 2095, 234

news2024/11/24 21:25:08

1. 力扣876:链表的中间节点

1.1 题目:

给你单链表的头结点 head ,请你找出并返回链表的中间结点。

如果有两个中间结点,则返回第二个中间结点。

示例 1:

输入:head = [1,2,3,4,5]
输出:[3,4,5]
解释:链表只有一个中间结点,值为 3 。

示例 2:

输入:head = [1,2,3,4,5,6]
输出:[4,5,6]
解释:该链表有两个中间结点,值分别为 3 和 4 ,返回第二个结点。

提示:

  • 链表的结点数范围是 [1, 100]
  • 1 <= Node.val <= 100

1.2 思考

快慢指针轻松解决。快指针每次走2步,慢指针每次走一步,快指针走完,慢指针走完了1 / 2,只是最后返回的时候需要判断一下,链表的长度是奇数还是偶数而已。简单判断一下就好了。

1.3 题解:

/**
 * Definition for singly-linked list.
 * public class ListNode {
 *     int val;
 *     ListNode next;
 *     ListNode() {}
 *     ListNode(int val) { this.val = val; }
 *     ListNode(int val, ListNode next) { this.val = val; this.next = next; }
 * }
 */
class Solution {
    public ListNode middleNode(ListNode head) {
        if(head == null) {
            return null;
        }
        ListNode fast = head;
        ListNode slow = head;
        while(fast.next != null && fast.next.next != null){
            fast = fast.next.next;
            slow = slow.next;
        }
        return fast.next == null ? slow : slow.next;
    }
}

2. 力扣2095:删除链表的中间节点

2.1 题目:

给你一个链表的头节点 head 。删除 链表的 中间节点 ,并返回修改后的链表的头节点 head 。

长度为 n 链表的中间节点是从头数起第 ⌊n / 2⌋ 个节点(下标从 0 开始),其中 ⌊x⌋ 表示小于或等于 x 的最大整数。

  • 对于 n = 1234 和 5 的情况,中间节点的下标分别是 0112 和 2 。

示例 1:

输入:head = [1,3,4,7,1,2,6]
输出:[1,3,4,1,2,6]
解释:
上图表示给出的链表。节点的下标分别标注在每个节点的下方。
由于 n = 7 ,值为 7 的节点 3 是中间节点,用红色标注。
返回结果为移除节点后的新链表。 

示例 2:

输入:head = [1,2,3,4]
输出:[1,2,4]
解释:
上图表示给出的链表。
对于 n = 4 ,值为 3 的节点 2 是中间节点,用红色标注。

示例 3:

输入:head = [2,1]
输出:[2]
解释:
上图表示给出的链表。
对于 n = 2 ,值为 1 的节点 1 是中间节点,用红色标注。
值为 2 的节点 0 是移除节点 1 后剩下的唯一一个节点。

提示:

  • 链表中节点的数目在范围 [1, 105] 内
  • 1 <= Node.val <= 105

2.2 思考

根据快慢指针的思想和两个案例,比较简单。

2.3 题解:

/**
 * Definition for singly-linked list.
 * public class ListNode {
 *     int val;
 *     ListNode next;
 *     ListNode() {}
 *     ListNode(int val) { this.val = val; }
 *     ListNode(int val, ListNode next) { this.val = val; this.next = next; }
 * }
 */
class Solution {
    public ListNode deleteMiddle(ListNode head) {
        if(head == null || head.next == null){
            return null;
        }
        // 头节点也可能被删除,所以需要哨兵节点
        ListNode dummy = new ListNode(10086, head);
        ListNode fast = dummy;
        ListNode slow = dummy;
        // 快指针和慢指针都从哨兵节点位置开始跑
        // 快指针每次走2步,慢指针每次走1步
        while(fast.next != null && fast.next.next != null){
            fast = fast.next.next;
            slow = slow.next;
        }
        // 分析两个测试,当fast停止步伐,slow都停在要删除节点的上一个节点
        slow.next = slow.next.next;
        return dummy.next;
    }
}

3. 力扣234:

3.1 题目:

给你一个单链表的头节点 head ,请你判断该链表是否为

回文链表

。如果是,返回 true ;否则,返回 false 。

示例 1:

输入:head = [1,2,2,1]
输出:true

示例 2:

输入:head = [1,2]
输出:false

提示:

  • 链表中节点数目在范围[1, 105] 内
  • 0 <= Node.val <= 9

进阶:你能否用 O(n) 时间复杂度和 O(1) 空间复杂度解决此题?

3.2 思路1:

将判断回文链表的问题转换为判断回文数组的问题。时间复杂度O(n), 空间复杂度O(n)。

还可以使用快慢指针法解决这个问题。从头节点开始,快指针每次走2步,慢指针每次走一步。快指针走完时,慢指针走到中间节点的上一个节点,然后将链表分为两个部分,将后半部分反转链表,然后就可以从两个链表的头节点开始比较。不如回文数组方法一根。

3.3 题解1:

/**
 * Definition for singly-linked list.
 * public class ListNode {
 *     int val;
 *     ListNode next;
 *     ListNode() {}
 *     ListNode(int val) { this.val = val; }
 *     ListNode(int val, ListNode next) { this.val = val; this.next = next; }
 * }
 */
class Solution {
    // 用栈的前提是把链表的中间节点首先从栈里给踢掉
    public boolean isPalindrome(ListNode head) {
        if(head == null || head.next == null){
            return true;
        }
        int n = 0;
        ListNode p = head;
        while(p != null){
            n++;
            p = p.next;
        }
        int[] arr = new int[n];
        p = head;
        int i = 0;
        while(p != null){
            arr[i++] = p.val;
            p = p.next;
        }
        // n表示数组长度,下面要表示索引,所以-1
        n--;
        for(i = 0; i < arr.length / 2; i++){
            if(arr[i] != arr[n--]){
                return false;
            }
        }
        return true;
    }
}

4. 力扣2130:链表最大孪生和

4.1 题目:

在一个大小为 n 且 n 为 偶数 的链表中,对于 0 <= i <= (n / 2) - 1 的 i ,第 i 个节点(下标从 0 开始)的孪生节点为第 (n-1-i) 个节点 。

  • 比方说,n = 4 那么节点 0 是节点 3 的孪生节点,节点 1 是节点 2 的孪生节点。这是长度为 n = 4 的链表中所有的孪生节点。

孪生和 定义为一个节点和它孪生节点两者值之和。

给你一个长度为偶数的链表的头节点 head ,请你返回链表的 最大孪生和 。

示例 1:

输入:head = [5,4,2,1]
输出:6
解释:
节点 0 和节点 1 分别是节点 3 和 2 的孪生节点。孪生和都为 6 。
链表中没有其他孪生节点。
所以,链表的最大孪生和是 6 。

示例 2:

输入:head = [4,2,2,3]
输出:7
解释:
链表中的孪生节点为:
- 节点 0 是节点 3 的孪生节点,孪生和为 4 + 3 = 7 。
- 节点 1 是节点 2 的孪生节点,孪生和为 2 + 2 = 4 。
所以,最大孪生和为 max(7, 4) = 7 。

示例 3:

输入:head = [1,100000]
输出:100001
解释:
链表中只有一对孪生节点,孪生和为 1 + 100000 = 100001 。

提示:

  • 链表的节点数目是 [2, 105] 中的 偶数 。
  • 1 <= Node.val <= 105

4.2 思考1

跟上题思路一模一样,转化成求解数组的问题,直接秒。

4.3 题解1:

/**
 * Definition for singly-linked list.
 * public class ListNode {
 *     int val;
 *     ListNode next;
 *     ListNode() {}
 *     ListNode(int val) { this.val = val; }
 *     ListNode(int val, ListNode next) { this.val = val; this.next = next; }
 * }
 */
class Solution {
    public int pairSum(ListNode head) {
        // 将链表的问题转变为数组的问题
        int n = 0;
        ListNode p = head;
        while(p != null){
            n++;
            p = p.next;
        }
        p = head;
        int[] arr = new int[n];
        int i = 0;
        while(p != null){
            arr[i++] = p.val;
            p = p.next;
        }
        n--;
        // 因为链表节点的值都是正的,所以可以设置为0
        int max = 0;
        for(i = 0; i < arr.length / 2; i++){
            max = Integer.max(max, arr[i]+arr[n--]);
        }
        return max;
    }
}

4.4: 思考2:

快慢指针法解决,时间上来说跟上面的方法差不多,但从空间上有了很大的进步。借用了一个哨兵节点的空间。而上一种方法却是申请了链表长度的数组。

4.5 题解2:

/**
 * Definition for singly-linked list.
 * public class ListNode {
 *     int val;
 *     ListNode next;
 *     ListNode() {}
 *     ListNode(int val) { this.val = val; }
 *     ListNode(int val, ListNode next) { this.val = val; this.next = next; }
 * }
 */
class Solution {
    public int pairSum(ListNode head) {
        // 快慢指针
        ListNode fast = head;
        ListNode slow = head;
        // 快指针一次走2步,慢指针一次走一步
        // 题目说了,链表的长度至少是2个
        // 所以就放心使用fast.next,不用担心fast空指针问题
        while(fast.next != null && fast.next.next != null){
            fast = fast.next.next;
            slow = slow.next;
        }
        //此时slow节点指向了中间节点的前一个节点
        // 记录slow指针的下一个节点
        ListNode next = slow.next;

        // 将一个链表一分为2
        slow.next = null;

        slow = next;
        // 此时slow才指向链表的中间节点

        // 下一步就是反转链表
        slow = traverse(slow);

        // 记录最大孪生和
        int max = 0;
        while(head != null){
            max = Integer.max(max, head.val + slow.val);
            head = head.next;
            slow = slow.next;
        }

        return max;
    }
    // 该方法返回了一个新的头节点
    private ListNode traverse(ListNode head){
        // 新链表的哨兵节点
        ListNode dummy = new ListNode(10086, null);
        // 头插法
        while(head != null){
            ListNode p = head.next;
            head.next = dummy.next;
            dummy.next = head;
            head = p;
        }
        return dummy.next;
    }
}

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

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

相关文章

pdf删除一页怎么删除?5种方法详细讲解,pdf删除页面实用技巧分享!

pdf删除一页怎么删除&#xff1f;从pdf文档中删除某页是一项非常实用的技术&#xff0c;特别是在需要编辑pdf文件时。在某些情况下&#xff0c;您可能需要删除页面以保护机密信息、去除不必要的内容&#xff0c;或者为了压缩pdf文件的大小。因此&#xff0c;掌握有效且简单的删…

搭建线上扭蛋机小程序,企业新的盈利模式?

近几年&#xff0c;随着市场对潮玩商品的需求不断增加&#xff0c;越来越多的资本入局到潮玩市场&#xff0c;因此&#xff0c;扭蛋机也开始在市场中持续走红&#xff0c;扭蛋机的各种创新玩法&#xff0c;也刺激着消费者的购买欲望&#xff0c;推动市场的发展。 扭蛋机的目标…

Java面试题下

Java面试题下 异常 Exception和Error有什么区别 所有的异常都有一个共同的祖先Throwable类。有两者子类: Exception:程序可以本身处理的异常&#xff0c;可通过catch来捕获。Exception 又可以分为 Checked Exception (受检查异常&#xff0c;必须处理) 和 Unchecked Excepti…

chapter14-集合——(List-HMap)——day18

目录 536-HMap阶段小结 537-HMAp底层机制 538-HMAP源码解读 539-HMap扩容树化触发 重点在于hashmap的扩容机制 它内部是k-v对方式存储数据&#xff0c;Hash$node类型&#xff1b;相同的key&#xff0c;就会覆盖 Hash$node类型实现了mapentry接口 扩容条件&#xff0c;看是否…

【在Linux世界中追寻伟大的One Piece】五种IO模型和阻塞IO

目录 1 -> 五种IO模型 1.1 -> 阻塞IO(Blocking IO) 1.2 -> 非阻塞IO(Non-blocking IO) 1.3 -> 信号驱动IO(Signal-Driven IO) 1.4 -> IO多路转接(IO Multiplexing) 1.5 -> 异步IO(Asynchronous IO) 2 -> 高级IO概念 2.1 -> 同步通信VS异步通信…

鸿蒙介绍、鸿蒙编程环境、基本组件、页面跳转学习

系列文章目录 第一章 鸿蒙介绍、鸿蒙编程环境、基本组件、页面跳转学习 文章目录 系列文章目录前言一、HarmonyOS基础1. 鸿蒙系统是什么&#xff1f;2. 鸿蒙系统的重要目录及文件 二、HarmonyOS编程介绍1. ArkTS编程语言介绍2. DevEco Studio编程环境介绍3. 关系介绍 三、使用…

ButterKnife:Android视图绑定的简化专家

在Android应用开发中&#xff0c;与UI组件的交互是不可或缺的一部分。然而&#xff0c;传统的视图绑定方式往往涉及大量的样板代码&#xff0c;这不仅增加了代码的复杂性&#xff0c;也使得维护变得更加困难。为了解决这一问题&#xff0c;Jake Wharton推出了ButterKnife&#…

【鸿蒙】HarmonyOS NEXT星河入门到实战2-ArkTS快速入门

目录 一、ArkTS基础快速入门 二、认识和存储数据 2.1 认识数据 2.2 存储数据&#xff08;变量、常量&#xff09; 2.2.1 变量 2.2.2 常量&#xff08;不可修改&#xff09; 三、数组 四、函数-Function 4.1 函数的基本使用 4.1.1 定义函数 4.1.2 调用函数 4.2 函数…

leetcode 2576.求出最多标记下标

2576.求出最多标记下标 题意&#xff1a; 解析&#xff1a; 数组长为 n n n&#xff0c;因为一次标记两个&#xff0c;所以数组中最多有 ⌊ n 2 ⌋ \lfloor \frac{n}{2}\rfloor ⌊2n​⌋ 对标记。 贪心的考虑&#xff0c;一个数 x 一定优先与满足 y ≥ 2 x y \ge 2x y≥2…

驱动(RK3588S)第十一课时:linux内核定时器和poll轮询

目录 学习目标一、内核的定时器1、定时器概念2、定时器的作用与分类3、定时器API函数1、初始化定时器核心结构体2、定时器核心结构体3、向内核注册定时器资源用于激活定时器4、删除定时器的资源5、这是改变定时器时间的函数&#xff0c;如果在指定的定时器(timer)没超时前调用&…

测评造假?Mistral首个多模态模型Pixtral 12B发布

测评造假&#xff1f;Mistral首个多模态模型Pixtral 12B发布&#xff01; 近日&#xff0c;法国人工智能&#xff08;AI&#xff09;初创公司Mistral于9月11日宣布推出其首款多模态AI大模型——Pixtral 12B&#xff0c;成功吸引了全球科技界的广泛关注。这款集图像与文本处理能…

IO流的使用

一、IO流的体系 二、代码应用 import java.io.*;public class Demo05 {public static void main(String[] args) throws IOException {copy1(); //1 使用原始的字节流按照一个一个字节的形式复制文件。copy2(); //2 使用原始的字节流按照字节数组的形式复制文件。copy3(); //3…

论文:AOP框架安全框架-系统架构师(六十六)

1详细论述安全架构设计中鉴别框架和访问控制框架设计内容&#xff0c;并论述鉴别框架和访问控制所面临的主要威胁&#xff0c;说明其危害。 解析&#xff1a; 鉴别框架有用户密码鉴别、生物特征鉴别和多因素鉴别。 用户密码鉴别可以采用验证登入的用户账号是否正确。 生物特…

SOMEIP_ETS_093: SD_Check_Reboot_Detection_separate_multicast_and_unicast

测试目的&#xff1a; 验证DUT&#xff08;Device Under Test&#xff09;能够检测到客户端在发送多播&#xff08;Multicast&#xff09;和单播&#xff08;Unicast&#xff09;时执行了重启。 描述 本测试用例旨在确保DUT能够区分客户端在多播和单播情况下的重启行为&…

刷题活动(旋转和翻转)

前两天打了CCPC网络赛&#xff08;让打老实了&#xff09;&#xff0c;现在认识到了刷题的重要性&#xff0c;于是我开创了这么个栏目&#xff0c;我们一起刷一下题。 还是在ACwing网站上刷题 旋转和翻转 首先&#xff0c;申一下题目&#xff0c;输入一个数字 n &#xff0c;来…

Linux | 进程控制(上):进程终止(strerror函数、errno宏、_exit() 与 exit())

文章目录 进程控制1、进程终止1.1进程常见退出方法退出码1.1.1 strerror函数 & errno宏1.1.1 _exit函数_exit和exit的区别结合现象分析&#xff1a; 进程控制 1、进程终止 1.1进程常见退出方法 进程退出场景 代码运行完毕&#xff0c;结果正确代码运行完毕&#xff0c;结…

计算机网络 数据链路层 3

以太网&#xff1a;采用CSMA/CD载波监听多路访问/冲突检测 基带总线局域网规范 以太网提供无连接&#xff0c;不可靠服务&#xff1a; 无连接&#xff1a;事先不必建立链路 不可靠&#xff1a;发送方的数据帧不进行编号&#xff0c;接收方接收信息后不向发送方发送ACK&#x…

深度神经网络DNN、RNN、RCNN及多种机器学习金融交易策略研究|附数据代码

全文链接&#xff1a;https://tecdat.cn/?p37668 原文出处&#xff1a;拓端数据部落公众号 分析师&#xff1a;Aijun Zhang 在当今的金融领域&#xff0c;量化交易正凭借其科学性和高效性逐渐成为主流投资方式之一。随着大数据技术的蓬勃发展&#xff0c;量化交易借助先进…

en造数据结构与算法 c#语言 数组实现队列很难???看我一击破之!!!

队列的特点就是先入先出 这回不像栈那样只需要瞄准最后一个坑了 你要入队的话&#xff0c;肯定要加到最后一个坑上&#xff0c;所以要守住最后一个坑 但是&#xff0c;你只有最后一个坑的标记还不行&#xff0c;因为出队你得退出第一个坑不是么 public class SimpleQueue<…

前端开发之迭代器模式

在前端开发中&#xff0c;设计模式是提升代码可读性、可扩展性和可维护性的关键。迭代器模式&#xff08;Iterator Pattern&#xff09;是行为型设计模式中的一种&#xff0c;能够让我们顺序访问一个集合中的元素&#xff0c;而不暴露其底层的结构。在 TypeScript 这样具有类型…