单链表初阶的两道基础题

news2024/9/20 14:37:02

初阶链表刷题

  • 翻转单链表(链接在末尾)
  • 链表的倒数第K个结点(链接在末尾)
    • 普通解法
    • 进阶解法

注意!!!

学习的是解题的思维!

翻转单链表(链接在末尾)

在这里插入图片描述
解题思路

如果给你一个1->2->3->4->5->null的链表,你需要将他改为5->4->3->2->1->null

迭代解法

我们先看图理解一下
在这里插入图片描述

  • 在第一步:我们要做的是将cur.next指向prev,但是当我的cur.next指向prev时就找不到了下一个结点。
  • 第二步:所以我们需要再定义一个指针去存储cur的下一个结点curNext=cur.next,保存下一个结点后,我们就可以让cur.next=prev了,接下来呢?
  • 第三步:我们需要做的就是让cur向右移,再让cur的next指向1这个结点,那么我们如何找到1这个结点呢?可以看到,我们的cur.next=null时,我们的cur正好是1这个结点,所以呢?我们就可以将cur赋值给prev:prev=cur,这样我们的prev就是1这个结点了(蓝框上部分)。
  • 第四步:因为curNext指针存了下一个结点的地址,所以我们可以直接让cur=curNext,这样我们的cur就是下一个结点了,经过这个操作之后再将curNext置为cur的后一个结点,也就是curNext=cur.next
  • 最后如蓝色框上部分

如果还有疑问,那就看下图
在这里插入图片描述

在遍历链表时,将当前节点的 cur.next 指针改为指向前一个节点。由于找不到前一个节点,因此必须事先存储其前一个节点。在更改引用之前,还需要存储后一个节点。最后返回新的头引用。

代码如下

class Solution {
    public ListNode reverseList(ListNode head) {
        //没有结点或者只有一个结点不需要逆置
        if (head==null||head.next==null) {
            return head;
        }
        ListNode prev=null;//记录前一个节点
        ListNode cur=head;//记录当前结点
        while (cur!=null) {
       	   //记录后一个结点
            ListNode curNext=cur.next;
            cur.next=prev;//当前结点的next指向前一个节点
            prev=cur;//将前一个结点移动到当前结点
            cur=curNext;//当前结点移动到下一个结点
        }
        //最后返回头节点
        return prev;
  }

复杂度分析:

  • 时间复杂度O(n),其中 N 是给定链表中的结点数目。
  • 空间复杂度O(1)

链表的倒数第K个结点(链接在末尾)

在这里插入图片描述

普通解法

很显然,求倒数第k个,可以转换成求正数第多少个呢?

我们来看图 在这里插入图片描述

我们总共有五个结点,如果需要找到倒数第3个,那就是正数第二个,也就是正数第n-k个,n为链表长度。

  • 第一步:遍历链表记录链表的长度
  • 第二步:用链表长度减去k,也就是n-k,求出正数第几位
  • 第三步:再次遍历链表,当循环n-k次后退出循环并返回

如图:

第一遍遍历

在这里插入图片描述

第二遍遍历

在这里插入图片描述
有了上面的过程,还需要考虑2个细节

  • k<=0 或者头结点最开始就为空,也就是没有节点
  • 节点的总个数 < K

代码如下

public class Solution {
    public ListNode FindKthToTail(ListNode head,int k) {
    //判断链表是不是为空,并且k不能小于等于0
        if (k<=0||head==null) {
            return null;
        }
        int n=0;//记录链表长度
        ListNode cur=head;
        while (cur!=null) {
            cur=cur.next;
            n++;
        }
        //当我的链表长度小于K说明K不在链表里面
        if(n<k) {
            return null;
        }
        cur=head;
        int size=n-k;//循环size次
        while(size>0) {
                cur=cur.next;
                size--;
        }
        return cur;
    }

n为链表的总长度,如果k总是在倒数第一个节点,那么此方法需要遍历链表2次,最坏时间复杂度为O(2*n)

复杂度分析:

  • 时间复杂度O(2*n) 其中 N 是给定链表中的结点数目。
  • 空间复杂度O(1)

进阶解法

定义一个快慢指针,先让快指针走K步,然后慢指针和快指针一起走,当快指针为null时,我们的慢指针就是倒数第K个数。

究竟上面的原理是如何推理出来的呢?

我们看图演示

在这里插入图片描述

第一步:先让快指针cur走K步,如图二
第二步:快指针,慢指针一起走,当快指针为null时退出循环
第三步:此时prev就是倒数第K个结点,返回prev就行

具体过程如下

在这里插入图片描述

使用如图的快慢指针,首先让快指针先行k步,然后让快慢指针每次同行一步,直到快指针指向空节点,慢指针就是倒数第K个节点

代码入下
注意注释!!!

public class Solution {
    public ListNode FindKthToTail(ListNode head,int k) {
		 //判断链表是不是为空,并且k不能小于等于0	
        if (k<=0||head==null) {
            return null;
        }
        int n=0;
        ListNode prev=head;
        ListNode cur=head;
        while (n<k) {
        //判断cur是否为null
        //如果为null返回null
            if (cur==null) {
               return null;
            }
            cur=cur.next;
            n++;
            //如果将if放到下面,
            //那么判断倒数第n个结点时就会出错,n是链表长度,
            //也就是查找链表头节点时会报错
            //if (cur==null) {
            // return null;
            // }
        }
        while (cur!=null) {
            prev=prev.next;
            cur=cur.next;
        }
        return prev;
    }
 }

作者总结:

有一个极端条件就是当倒数第K个结点是头节点时,不能将if放到我在代码注释的那个位置,如下图
在这里插入图片描述

如果在下面判断当cur == null时就退出,那么存在一种情况就是当我的cur==null时,我的prev就是倒数第5个结点符合条件,如果此时退出结果就会和预期结果不一样,所以我们将判断的条件写在了上面,那样的话,只有当我们的n没有符合条件时并且cur==null,才返回null,这样就避免了上面的错误

反转一个单链表:oj题

输入一个链表,输出该链表中倒数第k个结点oj题

下期预告:合并两个有序链表,判断回文链表,链表的相交结点

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

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

相关文章

Java#31(不可变集合,Stream流和方法引用)

目录 一.创建不可变集合: 不可以被修改的集合 1.List接口创建不可变集合 2.Set接口创建不可变集合 3.Map接口创建不可变集合 二.Stream流 1.如何获取Stream流? 2.Stream流的中间方法 3.Stream流终结方法 三.方法引用 1.引用静态方法 2.引用成员方法 3.引用结构方法…

【Java学习】JavaWeb---Request Response

文章目录1. Request1. 1Request继承体系1.2 Request获取请求数据1.3 Request请求转发2. Response2. 1 Response设置响应数据功能介绍2. 2 Response完成重定向2. 3 Response响应字符数据2. 4 Response响应字节数据1. Request 1. 1Request继承体系 1.2 Request获取请求数据 1.3 …

制造企业如何借数字化能力进军万亿国际市场?

历时七十余载&#xff0c;中国建立了世界最完整的现代工业体系&#xff0c;实现了从“制造大国”向“制造强国”的历史性跨越&#xff0c;同时&#xff0c;这片土地也孕育了全球最庞大的自动化物流系统。 报告显示&#xff0c;从2012到2021年&#xff0c;中国社会物流总额和社…

转行软件测试我后悔了

很多时候&#xff0c;都在想当初做的转行软件测试行业的决定是对的吗&#xff1f;现在后悔还来得及吗&#xff1f; 记得在求职的时候&#xff0c;面试官经常问我&#xff1a;“为什么要选择软件测试工作?” 而我也会经常说一堆自己有的没的优势去应付。 工作这么久了&#x…

anaconda迁移深度学习虚拟环境 and 在云服务器上配置

1 anaconda 虚拟环境操作 1、 查看虚拟环境 conda info -e2、 创建新的虚拟环境 conda create -n deeplearning_all pip python3.63、 激活新建的虚拟环境 Conda activate deeplearning_all2 环境中相关库的版本即安装说明&#xff08;这些库都是对应匹配的&#xff09; …

【git】git@github.com: Permission denied (publickey).报错问题

1、引发这个问题的情况 本媛开发&#xff0c;会经常性用到gitee & GitHub两个库 毕竟国内项目转战仓到gitee&#xff0c; 但是国外原框架还是GitHub居多 于是就出现连接pull-push经常性切换问题 这个报错是因为本地两个仓都有ssh公共私有密钥导致的 2、解决方案&#xff…

ECharts数据可视化(案例)

ECharts数据可视化 ECharts介绍 官网链接&#xff1a;https://echarts.apache.org/zh/index.html ECharts是一个使用 JavaScript 实现的开源可视化库&#xff0c;可以流畅的运行在 PC 和移动设备上&#xff0c;兼容当前绝大部分浏览器&#xff08;IE8/9/10/11&#xff0c;Ch…

华为高级技术专家多年经验分享微服务治理体系、架构及实践文档

前言 都说程序员工资高、待遇好&#xff0c; 2022 金九银十到了&#xff0c;你的小目标是 30K、40K&#xff0c;还是 16薪的 20K&#xff1f;作为一名 Java 开发工程师&#xff0c;当能力可以满足公司业务需求时&#xff0c;拿到超预期的 Offer 并不算难。然而&#xff0c;提升…

WEB前端网页设计 HTML网页代码 基础参数(一)

插入图片&#xff1a;<img src"./图像URL"> 整个HTML格式分为两大部分&#xff0c;一个是head部分&#xff0c;一个是body部分&#xff0c;其中head部分是用于书写网页样式&#xff0c;而body部分用于书写网页主题 标题标记 <h1></h1> ... <h6&…

生产型企业中采购管理系统的优势有哪些?

采购是生产型企业重要的环节&#xff0c;采购成本更是与企业利润息息相关。通常企业采购主要是取自于企业本身的需求&#xff0c;由于不同企业的需求不同&#xff0c;采购管理也就自然有所区别了。尤其是对于中小型企业而言&#xff0c;由于其本身规模有限&#xff0c; 生产资金…

[附源码]JAVA毕业设计健身房管理系统(系统+LW)

[附源码]JAVA毕业设计健身房管理系统&#xff08;系统LW&#xff09; 目运行 环境项配置&#xff1a; Jdk1.8 Tomcat8.5 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&a…

如何在Linux下 自定义/编写 一个守护进程

【摘要】本文主要讲述何为守护进程&#xff0c;以及如何自编或者利用现有程序将其伪装成守护进程。 何为守护进程&#xff1f; 正常情况下&#xff0c;当我们运行一个前台或后台进程时&#xff0c;一旦离开当前会话&#xff08;终端&#xff09;&#xff0c;那该会话中的所有…

BlockingQueue

网上看了好多文章将线程池的但是似乎都没的多少人会详细讲解里面的任务队列&#xff0c;所以只有自己动手学习其中的任务队列 BlockingQueue 要学习其中的任务队列就需要先学习BlockingQueue&#xff0c;Blocking是一个接口&#xff0c;其中主要的方法为 // 尝试往队尾添加元素…

m基于OFDM的OMP压缩感知信道估计算法误码率仿真,对比传统的LS,MMSE以及LMMSE信道估计性能

目录 1.算法描述 2.仿真效果预览 3.MATLAB核心程序 4.完整MATLAB 1.算法描述 正交频分复用技术(orthogonalfrequencydivisionmultiplexing,ofdm)应用在通信系统中可以有效抵抗码间干扰(inter-symbolinterference,isi)。同时&#xff0c;通过在符号间插入循环前缀(cyclicpre…

格利尔在北交所上市:市值突破9亿元,朱从利夫妇为实控人

12月2日&#xff0c;格利尔数码科技股份有限公司&#xff08;下称“格利尔”&#xff0c;BJ:831641&#xff09;在北京证券交易所上市。本次上市&#xff0c;格利尔公开发行1050万股&#xff0c;发行价格为9.60元/股&#xff0c;发行市盈率为23.32倍。 据了解&#xff0c;格利尔…

压缩文件7-Zip与WinRAR个人免费版在不同压缩等级下的对比

总结&#xff1a; 压缩率越高&#xff0c;压缩及解压时间相对更长&#xff0c;但传输时间越短&#xff0c;消耗流量越少。在各个压缩等级下&#xff0c;7-Zip均比RAR的压缩率更高&#xff0c;更能达到**“压缩”**的目的&#xff1b;而且相同参数下的压缩速度更快&#xff0c;…

基于增强蛇优化算法求解单目标优化问题附matlab代码

✅作者简介&#xff1a;热爱科研的Matlab仿真开发者&#xff0c;修心和技术同步精进&#xff0c;matlab项目合作可私信。 &#x1f34e;个人主页&#xff1a;Matlab科研工作室 &#x1f34a;个人信条&#xff1a;格物致知。 更多Matlab仿真内容点击&#x1f447; 智能优化算法 …

从源码分析vue3组件的生命周期

概览 借官网一张图充篇幅☺ 这张图展示了一个vue组件从开始渲染到卸载结束一整个生命周期经历的每个环节 但只罗列了选项式api生命周期钩子&#xff0c;没有将组合式api的生命周期钩子放进去 下面这个表格列出了所有选项式api生命周期钩子和组合式api生命周期钩子&#xff0c…

函数的节流和防抖?节流和防抖的区别及实现

一.防抖和节流的本质 本质上就是优化高频率执行代码的一种手段。 比如说&#xff1a;浏览器的scroll,keypress,mousemove,resize等事件在触发时&#xff0c;会不断的调用绑定在事件上的回调函数&#xff0c;会极大地浪费资源&#xff0c;降低前端的性能。 因此为了优化用户体…

【C++模板】非类型模板参数

目录什么是非类型模板参数&#xff1f;非类型的类模板参数非类型的函数模板参数非类型模板参数的局限性限制使用的场景支持使用的场景什么是非类型模板参数&#xff1f; 在函数模板和类模板中&#xff0c;模板参数并不仅仅可以当作类型&#xff0c;还可以当作普通值。当使用普通…