链表面试练习习题(Java)

news2024/11/15 18:50:12

1.

思路:

创建两个链表,一个用来记录小于x的结点,一个用来记录大于等于x的结点,然后遍历完原链表后,将小于x的链表和大于等于x的链表进行拼接即可

public class Partition {  
    public ListNode partition(ListNode pHead, int x) {  
        if (pHead == null || pHead.next == null) {  //如果链表为空或者只有一个结点
            return pHead;  
        }  
  
        ListNode beforeHead = new ListNode(0); // 哑节点,用于小于x的链表  
        ListNode before = beforeHead;  
        ListNode afterHead = new ListNode(0); // 哑节点,用于大于等于x的链表  
        ListNode after = afterHead;  
  
        while (pHead != null) {  
            if (pHead.val < x) {  
                before.next = pHead;  
                before = before.next;  
            } else {  
                after.next = pHead;  
                after = after.next;  
            }  
            pHead = pHead.next;  
        }  
  
        // 断开大于等于x的链表的尾部  
        after.next = null;  
        // 将小于x的链表连接到大于等于x的链表之前  
        before.next = afterHead.next;  
  
        return beforeHead.next;  
    }  
}

2.

思路:

首先要判断回文,必须要先找到中间结点,然后以中间结点为分界线,进行左右两部分判断,所以可以采用快慢指针找中间结点

因为是单向链表,右边这一部分很难从后往前走判断是否回文,所以我们要解决这个问题,可以想到的办法就是反转链表,将后面的结点的next指向它的前一个结点,相当于将中间结点后面的单向链表变成双向链表,同时慢指针走到最后一个结点

最后头结点往后走,慢指针往前走(只能用慢指针,因为当结点个数为偶数时,快指针会指向null)进行判断是否回文

public class PalindromeList {
    public boolean chkPalindrome(ListNode A) {
        // write code here
        //第一步:找到中间结点
        ListNode slow = A;
        ListNode fast = A;

        while (fast != null && fast.next != null) {
            slow = slow.next;
            fast = fast.next.next;
        }
        //第二步:反转中间结点后面的链表
        ListNode cur = slow.next;
        ListNode curN = cur.next;
        while (cur != null) {
            curN = cur.next;
            cur.next = slow;
            slow = cur;
            cur = curN;
        }
        //第三步:判读回文
        while (slow != A) {//当还没有相遇时
            if(A.next==slow&&A.val==slow.val){//偶数个结点回文判断结束条件
                return true;
            }
            if (slow.val != A.val) {//不是回文
                return false;
            }else{//继续判断
                slow=slow.next;
                A=A.next;
            }
        }
        return true;//奇数个结点回文判断结束条件
    }
}

3.

思路:

法一:一个链表作为参照,一个链表用来遍历,如果参照链表的这个结点经过另外那个链表遍历后没有发现相交,参照链表的结点往后,另外那个链表继续遍历,如此类推

法二:如果相交,那么相交之后的链表长度是相同的,所以两个链表长度的差值就等于相交前面的链表长度差值,因此只需要求出两个链表的长度,相减得出差值,让较长的链表先走差值步,然后再以相同的速度往后走,则相等时相交

代码(法一):

public class Solution {
    public ListNode getIntersectionNode(ListNode headA, ListNode headB) {
        ListNode a=headA;
        ListNode b=headB;
        while(a!=null){//每次遍历一个a链表的结点
            while(b!=null){//遍历全部b链表的结点
                if(b==a){//如果相交
                    return b;
                }else{
                    b=b.next;//继续找
                }
            }
            b=headB;//从头开始
            a=a.next;//找下一个结点
        }
        return null;//没有相交
    }
}

代码(法二):

public class Solution {
    public ListNode getIntersectionNode(ListNode headA, ListNode headB) {
        ListNode a=headA;
        ListNode b=headB;
        int len1=0;//a链表的长度
        int len2=0;//b链表的长度
        int len=0;//差值
        while(a!=null){//求a链表的长度
            len1++;
            a=a.next;
        }
        while(b!=null){//求b链表的长度
            len2++;
            b=b.next;
        }
        if(len1>len2){//如果是a链表更长
            len=len1-len2;
            a=headA;//a指向更长的那条链表
            b=headB;//b指向更短的那条链表
        }else{//如果是b链表更长
            len=len2-len1;
            a=headB;//a指向更长的那条链表
            b=headA;//b指向更短的那条链表
        }
        while(len!=0){//让长链表先走差值步
            len--;
            a=a.next;
        }
        while(a!=b){//如果还没有相交
            a=a.next;
            b=b.next;
        }
        if(a==null){//没有相交
            return null;
        }
        return a;//相交点
    }
}

4.

思路:

如果没有环的话,那么一定会有一个尽头,即一定会走到null,如果有环的话,那么将永远没有尽头,即不存在null,这个环可以看作是一个圆形操场,就变为简单的追及相遇这个数学问题,所以需要快慢指针,如果有环,那么快指针一定和慢指针相遇

public class Solution {
    public boolean hasCycle(ListNode head) {
        if(head==null){//如果是空链表
            return false;
        }
        ListNode slow=head;
        ListNode fast=head;
        while(fast!=null&&fast.next!=null){//如果走到了尽头
            slow=slow.next;
            fast=fast.next.next;
            if(slow==fast){//如果相遇,说明有环
                break;
            }
        }
        if(fast==null||fast.next==null){//如果是因为走到尽头而跳出的循环
            return false;//没有环
        }
        return true;//如果是因为相遇而跳出的循环,则有环
    }
}

5.

思路:

首先判断是否有环,跟上面一题一样,接下来,如果有环,求出环的开始结点,还是快慢指针,快指针一次走两步,慢指针一次走一步,用两个公式表示快慢指针走的路程,其中C是环的长度,K是相遇点距离环的开始结点的长度,X表示头结点到环的开始结点的长度

1.fast走的路程S1:X+NC+C-K

2.slow走的路程S2:X+C-K

并且S1=2S2

所以

X+NC+C-K=2*(X+C-K)

X+NC+C-K=2X+2C-2K

X=(N-1)C+K

由该公式可得,从头结点到环的开始结点等于从相遇点走N-1圈再走到环的开始结点,所以让slow从头结点开始,fast从相遇点开始,以相同的速度开始走,那么再次相遇的位置就是环的开始结点

public class Solution {
    public ListNode detectCycle(ListNode head) {
        if(head==null){//如果是空链表
            return null;
        }
        ListNode slow=head;
        ListNode fast=head;
        while(fast!=null&&fast.next!=null){//走到尽头
            slow=slow.next;
            fast=fast.next.next;
            if(slow==fast){//如果相遇
                break;
            }
        }
        if(fast==null||fast.next==null){//如果是因为走到尽头,则无环
            return null;
        }
        //因为相遇,则有环
        slow=head;//从头走到环的开始结点
        while(slow!=fast){//走到相遇
            slow=slow.next;
            fast=fast.next;
        }
        return slow;//返回相遇点(即环的开始结点)
    }
}

6.

思路:

首先先判断特殊情况:两个链表都为空,或者其中一个链表为空,则返回null

然后是正常情况,因为两个链表都是升序的,所以先比较两个链表的头结点,较小的那个结点则一定为两个链表中最小的,所以该结点为合并后新的头结点,然后用一个指针cur去接收两个链表较小的结点,尾插在新的头结点后面,如此类推

class Solution {
    public ListNode mergeTwoLists(ListNode list1, ListNode list2) {
        ListNode newH=null;//合并后的头结点
        if(list1==null&&list2==null){//两个链表都为空
            return null;
        }
        if(list1==null){//其中一个链表为空
            return list2;
        }
        if(list2==null){//另外一个链表为空
            return list1;
        }
        ListNode a=list1;
        ListNode b=list2;
        if(a.val<b.val){
            newH=a;//说明合并后的新头结点为a链表的头结点
            a=a.next;
        }else{
            newH=b;//说明合并后的新头结点为b链表的头结点
            b=b.next;
        }
        ListNode cur=newH;//用来遍历链表
        while(a!=null&&b!=null){//当两个链表都没有走完
            if(a.val<b.val){//a的小于b,则插a
                cur.next=a;
                cur=cur.next;
                a=a.next;
            }else{//a的大于等于b,则插b
                cur.next=b;
                cur=cur.next;
                b=b.next;
            }
        }
        while(a!=null){//当b链表走完而a链表没有走完,将a链表全部尾插
            cur.next=a;
            cur=cur.next;
            a=a.next;
        }
        while(b!=null){//当a链表走完而b链表没有走完,将b链表全部尾插
            cur.next=b;
            cur=cur.next;
            b=b.next;
        }
        return newH;//返回合并后的新头结点
    }
}

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

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

相关文章

【JVM】JVM调优练习-随笔

JVM实战笔记-随笔 前言字节码如何查看字节码文件jclasslibJavapArthasArthurs监控面板Arthus查看字节码信息 内存调优内存溢出的常见场景解决内存溢出发现问题Top命令VisualVMArthas使用案例 Prometheus Grafana案例 堆内存情况对比内存泄漏的原因:代码中的内存泄漏并发请求问…

自动化测试中如何应对网页弹窗的挑战!

在自动化测试中&#xff0c;网页弹窗的出现常常成为测试流程中的一个难点。无论是警告框、确认框、提示框&#xff0c;还是更复杂的模态对话框&#xff0c;都可能中断测试脚本的正常执行&#xff0c;导致测试结果的不确定性。本文将探讨几种有效的方法来应对网页弹窗的挑战&…

IDEA中内敛局部变量对话窗是什么?

个人名片 &#x1f393;作者简介&#xff1a;java领域优质创作者 &#x1f310;个人主页&#xff1a;码农阿豪 &#x1f4de;工作室&#xff1a;新空间代码工作室&#xff08;提供各种软件服务&#xff09; &#x1f48c;个人邮箱&#xff1a;[2435024119qq.com] &#x1f4f1…

【Hive SQL 每日一题】找出各个商品销售额的中位数

文章目录 测试数据需求说明需求实现方法1 —— 升序计算法方法2 —— 正反排序法 补充 测试数据 -- 创建 orders 表 DROP TABLE IF EXISTS orders; CREATE TABLE orders (order_id INT,product_id INT,order_date STRING,amount DOUBLE );-- 插入 orders 数据 INSERT INTO ord…

【Linux】Linux环境设置环境变量操作步骤

Linux环境设置环境变量操作步骤 在一些开发过程中本地调试经常需要依赖环境变量的参数&#xff0c;但是怎么设置对小白来说有点困难&#xff0c;今天就介绍下具体的操作步骤&#xff0c;跟着实战去学习&#xff0c;更好的检验自己的技术水平&#xff0c;做技术还是那句话&…

HTML+JS+CSS计算练习

可填 题目数量 数字范围 计算符号 题目做完后会弹窗提示正确率、用时 效果图 源代码在图片后面 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevic…

读AI新生:破解人机共存密码笔记18读后总结与感想兼导读

1. 基本信息 读AI新生&#xff1a;破解人机共存密码 斯图尔特罗素 著 中信出版社,2020年8月出版 1.1. 读薄率 书籍总字数301千字&#xff0c;笔记总字数44614字。 读薄率44614301000≈14.8% 1.2. 读厚方向 千脑智能 脑机穿越 未来呼啸而来 虚拟人 AI3.0 新机器人 …

月亮和Pandas -- 开源项目的挑战与机遇

毛姆的《月亮和六便士》讲述了这样一个故事&#xff0c;一个富有的英国股票经纪人&#xff0c;思特里克兰德&#xff0c;为了追求艺术和美&#xff0c;放弃自己的生活和家庭&#xff0c;来到巴黎&#xff0c;以艺术家的身份过着贫穷却目中无人的充足的生活&#xff0c;最终留下…

芯科科技第五届物联网开发者大会走进世界各地,巡回开启注册

中国&#xff0c;北京 – 2024年7月18日 – 致力于以安全、智能无线连接技术&#xff0c;建立更互联世界的全球领导厂商Silicon Labs&#xff08;亦称“芯科科技”&#xff0c;NASDAQ&#xff1a;SLAB&#xff09;今日宣布&#xff0c;其2024年Works With开发者大会现正开放注册…

一文带你摸清SaaS服务与开源知识库的区别

在当今数字化转型的浪潮中&#xff0c;SaaS服务与开源知识库作为推动行业进步的两股重要力量&#xff0c;在多个方面展现出显著的区别。SaaS以其即用即付的便捷性、持续更新维护的优势&#xff0c;成为现代企业提升效率的首选&#xff1b;而开源知识库则凭借开放共享、自由定制…

SpringMVC注解全解析:构建高效Web应用的终极指南 (上)

SpringMVC 是一个强大的 Web 框架&#xff0c;广泛应用于 Java Web 开发中。它通过注解简化了配置&#xff0c;增强了代码的可读性。本文将全面解析 SpringMVC 中常用的注解及其用法&#xff0c;帮助你构建高效的 Web 应用。 一. MVC介绍 MVC 是 Model View Controller 的缩写…

项目实用linux 操作详解-轻松玩转linux

我之前写过完整的linux系统详解介绍&#xff1a; LInux操作详解一&#xff1a;vmware安装linux系统以及网络配置 LInux操作详解二&#xff1a;linux的目录结构 LInux操作详解三&#xff1a;linux实际操作及远程登录 LInux操作详解四&#xff1a;linux的vi和vim编辑器 LInux操作…

LeetCode 3112.访问消失节点的最少时间:单源最短路的Dijkstra算法

【LetMeFly】3112.访问消失节点的最少时间&#xff1a;单源最短路的Dijkstra算法 力扣题目链接&#xff1a;https://leetcode.cn/problems/minimum-time-to-visit-disappearing-nodes/ 给你一个二维数组 edges 表示一个 n 个点的无向图&#xff0c;其中 edges[i] [ui, vi, l…

【JavaEE】CAS原理实现 + 常见应用

本文基于jdk8 参考&#xff1a; 黑马程序员深入学习Java并发编程&#xff0c;JUC并发编程全套教程_哔哩哔哩_bilibili CAS原理 CAS&#xff1a;比较和交换(设置) Compare And Swap(Set)。当A的值为5的时候&#xff0c;给A设置值为10。这里涉及到的比较和设置值的操作是原子的…

SAP代理商哲讯智能携手合肥企业,共塑SAP系统开发新篇章

在数字化转型的浪潮中&#xff0c;SAP开发作为企业信息化建设的重要一环&#xff0c;对于提升企业的运营效率和市场竞争力具有至关重要的作用。合肥&#xff0c;作为中国东部地区的重要城市之一&#xff0c;拥有众多企业正积极探索SAP开发的可能性。哲讯智能科技有限公司&#…

【昇思25天学习打卡营第25天 | 基于MindSpore通过GPT实现情感分类】

学习心得&#xff1a;基于MindSpore通过GPT实现情感分类 摘要 本文通过一个具体的实验案例&#xff0c;详细阐述了如何使用华为的MindSpore框架结合GPT模型来实现情感分类任务。从环境配置到模型训练&#xff0c;再到评估和测试&#xff0c;整个流程清晰、系统&#xff0c;为…

第三篇 Vue项目目录结构介绍

1、最外层目录结构 passagerFrontPage ├── .vscode //vscode配置&#xff0c;不用理会 ├── node_modules //项目依赖&#xff0c;npm install命令执行后自动生成 ├── public //公共资源存放 ├── src //源码 ├── tests //选装&#xff1a;测试模块 ├── .git…

华为云安全事件深度剖析与防范策略

华为云安全事件深度剖析与防范策略 引言 随着云计算技术的飞速发展&#xff0c;企业越来越依赖云服务来提升业务效率和创新能力。然而&#xff0c;云服务在带来便利的同时&#xff0c;也伴随着一系列复杂的安全挑战。华为云作为国内领先的云服务提供商&#xff0c;其安全性直接…

Linux网络——TcpServer

一、UDP 与 TCP 在现实生活中&#xff0c;Udp 类似于发传单&#xff0c;Tcp 类似于邮局的挂号信服务。 1.1 UDP&#xff08;用户数据报协议&#xff09; 无连接&#xff1a;发放传单时&#xff0c;你不需要提前和接受传单的人建立联系&#xff0c;直接把传单发出去。不可靠&…

Ubantu 使用 docker 配置 + 远程部署 + 远程开发

大家好我是苏麟 , Ubantu 一些配置 . 视频 : 服务器很贵&#xff1f;搞台虚拟机玩玩&#xff01;保姆级 Linux 远程开发教程_哔哩哔哩_bilibili Docker安装及配置 安装命令 : sudo apt install docker.io 查看版本号 : docker -v 查看虚拟机地址命令 : ifconfig 虚拟机地址 或…