Leetcode 206 反转链表

反转链表

      • 准备工作
        • 1)ListNode基本结构
        • 2)初始化ListNode集合
      • 解法一:遍历创建新节点
      • 解法二:两组List,面向对象操作
      • 解法三:递归调用
      • 解法四:直接移动
      • 解法五:解法二的面向过程

Leetcode 206 反转链表

准备工作

1)ListNode基本结构
public class ListNode {
    public int val;
    public ListNode next;

    public ListNode(int val, ListNode next) {
        this.val = val;
        this.next = next;
    }

    @Override
    public String toString() {
        ListNode node = this;
        StringBuilder sb = new StringBuilder();
        while (node != null) {
            sb.append(node.val).append("->");
            node = node.next;
        }
        sb.append("NULL");
        return sb.toString();
    }
}
2)初始化ListNode集合
public class ListNodeInit {
    public static ListNode getInitList() {
        ListNode tailf = new ListNode(5, null);
        ListNode node4 = new ListNode(4, tailf);
        ListNode node3 = new ListNode(3, node4);
        ListNode node2 = new ListNode(2, node3);
        return new ListNode(1, node2);
    }
}



解法一:遍历创建新节点

新增一个ListNode newList集合,然后直接遍历目标ListNode initList集合,每遍历一个节点就创建一个新节点,并且将新节点添加到我们newList中

public class 反转链表1 {
    public static void main(String[] args) {
        ListNode initList = ListNodeInit.getInitList();
        System.out.println(initList);
        System.out.println("-----反转后-----");
        System.out.println(reverseList(initList));
    }

    private static ListNode reverseList(ListNode initList) {
        ListNode newList = null;
        ListNode p = initList;
        // 遍历到原链表的节点
        while (p != null) {
            int val = p.val;

            // 创建一个新节点,新节点的下一个节点是之前的数据(将新值添加到头部)
            newList = new ListNode(val, newList);
            p = p.next;
        }
        return newList;
    }
}



解法二:两组List,面向对象操作

构建一个包装类List集合,用于封装addFirst、removeFirst方法,思路为移除oldList头部节点,然后添加到新的newList的头部

public class 反转链表2 {

    public static void main(String[] args) {
        ListNode initList = ListNodeInit.getInitList();
        System.out.println(initList);
        System.out.println("-----反转后-----");
        System.out.println(reverseList(initList));
    }

    private static ListNode reverseList(ListNode initList) {
        List oldList = new List(initList);
        List newList = new List(null);

        while (true) {
            // 原集合不停移除头部元素
            ListNode removedNode = oldList.removeFirst();
            if (removedNode == null) {
                break;
            }
            // 新集合不停添加到头部(类比栈)
            newList.addFirst(removedNode);
        }
        return newList.head;
    }

    static class List {
        ListNode head;

        public List(ListNode head) {
            this.head = head;
        }

        /**
         * 向头部添加节点
         */
        public void addFirst(ListNode node) {
            node.next = head;
            head = node;
        }

        /**
         * 向尾部添加节点
         */
        public ListNode removeFirst() {
            ListNode removedList = head;
            if (head != null) {
                head = head.next;
            }
            return removedList;
        }
    }
}



解法三:递归调用

利用递归后续遍历的回溯,能够反向拿到每一个元素的特点,不断的向initList尾部追加我们需要的逆序链表

public class 反转链表3 {

    public static void main(String[] args) {
        ListNode initList = ListNodeInit.getInitList();
        System.out.println(initList);
        System.out.println("-----反转后-----");
        System.out.println(reverseList(initList));
    }

    private static ListNode reverseList(ListNode initList) {
        if (initList == null || initList.next == null) {
            return initList;
        }
        // initList.next.val最多为5,因为5.next.val为null。即initList.val最多为4
        ListNode node = reverseList(initList.next);

        // 第一次调用时:
        // 5 -> 4,3,2,1  &&  4 -> null
        // 第二次调用时:
        // 4 -> 3,2,1    &&  3 -> null
        // 第三次调用时:
        // 3 -> 2,1      &&  2 -> null
        // 此处递归为后续遍历,可获取5,4,3,2,1顺序的值。
        // 步骤1)拿到node节点后,将当前node和其head部分分开,将node的head部分的上一个节点赋值到node的next,
        // 步骤2)且同时切断node和其head部分之间的关联
        //
        // 补充:前面提到initList.next.val最多为5,所以initList.next.next就是5的下一个节点
        initList.next.next = initList;
        initList.next = null;
        return node;
    }
}



解法四:直接移动

有点像选择排序,对两个元素进行互换位置,只不过此处为链表,难度大于数组。通过指针的关联关系,硬操作集合本身,每拿到一个节点,都将节点移动到头部

执行流程:

指针移动效果:

public class 反转链表4 {
    public static void main(String[] args) {
        ListNode initList = ListNodeInit.getInitList();
        System.out.println(initList);
        System.out.println("-----反转后-----");
        System.out.println(reverseList(initList));
    }

    private static ListNode reverseList(ListNode o1) {
        if (o1 == null || o1.next == null) {
            return o1;
        }
        // 1、设置o1、o2、n1三者之间的关系
        ListNode o2 = o1.next;
        ListNode n1 = o1;

        while (o2 != null) {
            // 2、断开o2节点:o1的下一个节点本来是指向o2,现在直接修改为指向o2的下一个节点,所以能够断开
            o1.next = o2.next;
            // 3、o2添加到头部:o2的下一个节点是n1节点,n1表示新节点的头部
            o2.next = n1;
            // 4、修正n1的位置:在赋初值的时候,n1代表头部节点,由于上一步头部添加一个一个o2,所以n1不是头部,此处重新指向到头部
            n1 = o2;
            // 5、修正o2的位置:由于前面o2节点执行到了头部,此处将o2节点重新指向下一个需要处理的节点,即o1的下一个
            o2 = o1.next;
        }
        return n1;
    }
}



解法五:解法二的面向过程

与解法二思想相同,解法二为面向对象,此处为面向过程;
与解法四的区别为此处将原来的List集合一分为二,移动节点位置本身,没有太大的区别

public class 反转链表5 {
    public static void main(String[] args) {
        ListNode initList = ListNodeInit.getInitList();
        System.out.println(initList);
        System.out.println("-----反转后-----");
        System.out.println(reverseList(initList));
    }

    private static ListNode reverseList(ListNode initList) {
        if (initList == null || initList.next == null) {
            return initList;
        }

        // 1、初始化o1、o2、n1三者之间的关系
        ListNode n1 = null;
        ListNode o1 = initList;
        ListNode o2 = null;

        while (o1 != null) {
            // 2、标记处理节点的下一个节点:用于标记剩余待处理节点的头节点
            o2 = o1.next;
            // 3、构建新list:o1表示要处理的节点,n1为初始化节点,此处将初始化节点和在处理的o1节点进行关联
            o1.next = n1;
            // 4、修正新list的头节点:经过上面的赋值,n1位置已经向后移动一个,此处调整
            n1 = o1;
            // 5、修正待处理list的头节点:经过上面的赋值,此时o1指向新集合的头部,此时借助前面的o2,将o1的位置修正为待处理list的头部
            o1 = o2;
        }
        return n1;
    }
}

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

spring cloud activiti 审批流的用法

demo的搭建及使用 1、创建activiti审批流需要安装bpmn插件,新的idea版本支持的这个bpmn插件只有下图这个,并不好用,所以我这里使用eclipse来创建bpmn流程 eclipse的连接如下:链接:https://pan.baidu.com/s/1mSoKprN-…

基于Java SSM框架实现药品销售系统项目【项目源码+论文说明】

基于java的SSM框架实现药品销售系统演示 摘要 本论文主要论述了如何使用JAVA语言开发一个药品销售系统 ,本系统将严格按照软件开发流程进行各个阶段的工作,采用B/S架构,面向对象编程思想进行项目开发。在引言中,作者将论述药品销…

原生table样式

HTML <div><table style"width: 100%;"><thead><tr><th style"width:25%;">董事会</th><th style"width:25%;">监事会</th><th style"width:25%;">股东</th><th sty…

【Vue】前端项目引入阿里图标

【Vue&React】前端项目引入阿里图标 1、登录自己的iconfont-阿里巴巴矢量图标库&#xff0c;把需要的图标加入到自己的项目中去&#xff1b;2、加入并进入到项目中去选择Font class 并下载到本地3、得到的文件夹如下4. 把红框中的部分粘贴到自己的项目中&#xff08; stati…

CRG设计之复位

1. 前言 CRG(Clock and Reset Generation&#xff0c;时钟复位生成模块) 模块扮演着关键角色。这个模块负责为整个系统提供稳定可靠的时钟信号&#xff0c;同时在系统上电或出现故障时生成复位信号&#xff0c;确保各个模块按预期运行。简而言之&#xff0c;CRG模块就像是SoC系…

nginx无法启动,win10占用80端口 (注册表方式解决)

参考&#xff1a;https://blog.csdn.net/qq_39523111/article/details/128853509 改为4 重启后 不再占用 pid 不是4了 已经变为nginx了 改为0 没起作用

蓝桥杯2024/1/26笔记-----基于PCF8591的电压采集装置

功能实现要求&#xff1a; 每次建好工程文件夹&#xff0c;里边包含User&#xff08;放工程文件&#xff0c;mian.c&#xff0c;可以在这里写如同我这个文章的文本文档&#xff09;、Driver&#xff08;存放底层文件如Led.c&#xff0c;Led.h等&#xff09; 新建的工程先搭建框…

STM32——中断系统和外部中断EXTI

一、中断 1.1中断系统 中断系统是管理和执行中断的逻辑结构&#xff1b; 1.2中断 系统在执行主程序过程中&#xff0c;出现了特定的触发条件&#xff08;触发源&#xff09;&#xff0c;系统停止执行当前程序&#xff0c;转而去执行中断程序&#xff0c;执行完毕后&#xf…

windows系统下启动redis命令

windows系统下启动redis命令 进入redis安装目录 cd redis 输入 redis-server.exe redis.windows.conf 启动redis命令&#xff0c;看是否成功 可能会启动失败&#xff0c;报[1696] 30 Jan 09:46:07.518 # Creating Server TCP listening socket 127.0.0.1:6379: bind: No erro…

云计算底层技术、磁盘技术揭秘虚拟化管理、公有云概述

查看本机是否具备虚拟化支持 硬件辅助虚拟化 处理器里打开 虚拟化Inter VT-x/EPT 或AMD-V 构建虚拟化平台工具软件包 yum 与 dnf Yum和DNF都是用于管理Linux系统中的软件包的工具&#xff0c;但它们在许多方面存在一些差异。以下是一些可能的区别&#xff1a; 依赖解…

运行VUE提示找不到模块validate-engines.js...

原来好好的&#xff0c;突然提示找不到模块validate-engines.js&#xff0c;CMD命令行输入npm -v不是内部或外部命令&#xff0c;node -v可以查看到版本号。 解决&#xff1a; 1. 卸载nodejs&#xff0c;重新下载安装文件&#xff1a;下载nodejs 2. 到目录&#xff1a;C:\Us…

成功解决AttributeError: ‘str‘ object has no attribute ‘decode‘

成功解决AttributeError: ‘str’ object has no attribute ‘decode’. &#x1f335;文章目录&#x1f335; &#x1f333;引言&#x1f333;&#x1f333;报错分析及解决方案&#x1f333;&#x1f333;参考文章&#x1f333;&#x1f333;结尾&#x1f333; &#x1f333;引…

Chiplet,汽车“芯”风向

异构集成、高速互联、算力灵活可扩展正在成为新一轮汽车芯片竞争的焦点。尤其是随着以ChatGPT为代表的大数据、大模型产品在车端的落地&#xff0c;对于芯片的要求还在持续提升。 本周&#xff0c;12家日本汽车制造商&#xff08;包括丰田、日产、本田等&#xff09;、零部件制…

Redis 面试题 | 20.精选Redis高频面试题

&#x1f90d; 前端开发工程师、技术日更博主、已过CET6 &#x1f368; 阿珊和她的猫_CSDN博客专家、23年度博客之星前端领域TOP1 &#x1f560; 牛客高级专题作者、打造专栏《前端面试必备》 、《2024面试高频手撕题》 &#x1f35a; 蓝桥云课签约作者、上架课程《Vue.js 和 E…

三、防御保护---防火墙安全策略篇

三、防御保护---防火墙安全策略篇 一、什么是安全策略二、安全策略的组成1.匹配条件2.动作3.策略标识 三、防火墙的状态检测和会话表1.会话表2.状态检测技术 四、ASPF--隐形通道五、用户认证1.用户认证的分类2.认证方式3.认证策略4.认证域 一、什么是安全策略 传统的包过滤防火…

ORBSLAM3 运行流程 以rgbd_tum.cc函数为例进行分析

一、运行 使用的是D435i相机自己录制的数据。 运行命令&#xff1a; ./Examples/RGB-D/rgbd_tum /opt/vslam/ORB_SLAM3_detailed_comments-dense_map_new/Vocabulary/ORBvoc.txt /opt/vslam/ORB_SLAM3_detailed_comments-dense_map_new/Examples/RGB-D/TUM1.yaml /opt/vsl…

医美诊疗前后要注意的八大诀窍

【记者许家源/综合报导】 随着年龄的增长&#xff0c;许多人都想保持年轻美丽&#xff0c;因此寻求医美诊疗的帮助。然而&#xff0c;进入医美诊所后&#xff0c;你可能会发现&#xff0c;想要打肉毒、除毛等&#xff0c;实际花费和广告中的金额相差甚远。为了避免上当受骗&am…

C# 使用WMI监听进程的启动和关闭

写在前面 Windows Management Instrumentation&#xff08;WMI&#xff09;是用于管理基于 Windows 操作系统的数据和操作的基础结构。具体的API可以查看 WMI编程手册。 WMIC 是WMI的命令行管理工具&#xff0c;使用 WMIC&#xff0c;不但可以管理本地计算机&#xff0c;还可…

Layui + Echarts 5.0

Layui 怎么整合最新版本的 Echarts 5.0&#xff0c;Echarts 4 升级到 5后&#xff0c;有了很大改变&#xff0c;新的配置项4是无法兼容的&#xff0c;所以想要使用新的功能&#xff0c;都需要升级&#xff01; 新建一个echarts.js文件 layui.define(function (exports) {// 这…

【教程】iOS如何抓取HTTP和HTTPS数据包经验分享

&#x1f4f1; 在日常的App开发和研发调研中&#xff0c;对各类App进行深入的研究分析时&#xff0c;我们需要借助专业的抓包应用来协助工作。本文将介绍如何使用iOS手机抓包工具来获取HTTP和HTTPS数据包&#xff0c;并推荐一款实用的抓包应用——克魔助手&#xff0c;希望能够…