【算法通关村】链表反转经典问题解析

🚩本文已收录至算法学习之旅

一.基础反转

我们通常有两种方法反转链表,一种是直接操作链表实现反转操作,一种是建立虚拟头节点辅助实现反转操作

力扣习题链接:206. 反转链表
在这里插入图片描述

(1) 直接操作实现反转

我们需要一个变量pre来保存当前节点的上一个节点,否则无法进行反转操作也就是指向上一个节点。我们还需要一个节点next来保存当前节点的下一个节点,因为我们一旦操作当前节点的指针域后将会丢失下一个节点的地址。知道需要两个变量进行辅助,接下来我们就可以严格按照一定的顺序来反转指针的。1.先记录当前节点的下一个指针(不记录将会在下一步操作中丢失) 2.令当前节点指向前一个节点(如此我们便完成了这两个节点的反转操作,可以开始移动指针进行下两个节点之间的操作) 3.令pre指针指向当前指针(移动后当前指针变成了前一个指针,用于后续节点的反转)4. 将当前指针移向下一个指针(正是我们通过next变量保存了才能找到当前节点的下一个指针,否则就在第2步中丢失了后续节点)5.如此往复直到当前指针移动到null,我们也完成了反转操作。

	public ListNode reverseList(ListNode head) {
        // 记录前一个节点
        ListNode prev = null;
        ListNode curr = head;
        while (curr != null) {
            // 记录后一个节点
            ListNode next = curr.next;
			 // 令当前节点指向前一个节点
            curr.next = prev;
            // 保存当前节点
            prev = curr;
            // 移动到下一个节点
            curr = next;
        }
        return prev;
    }

强烈建议各位手动模拟,注意链表节点与节点之间是如何进行链接的,体会每个变量的作用,示意图如下:
在这里插入图片描述

(2) 虚拟节点辅助实现反转

在上一篇中,我们发现处理头节点与中间节点和尾部节点方法不一致,如果单独处理则比较麻烦,因此我们可以先建立一个虚拟头节点并且指向链表头节点head,这样我们的操作便统一了。通过使用虚拟节点辅助实现反转,我们可以每次从旧的链表拆下来一个结点接到ans后面,然后将其他线调整好就可以了。

public static ListNode reverseList(ListNode head) {
    ListNode ans = new ListNode(-1);
    ListNode cur = head;
    while (cur != null) {
        ListNode next = cur.next;
        cur.next = ans.next;
        ans.next = cur;
        cur = next;
    }
    return ans.next;
}

与直接反转类似,强烈建议各位手动模拟,注意链表节点与节点之间是如何进行链接的,体会每个变量的作用,示意图如下:

在这里插入图片描述

二.经典问题分析

(1) 指定区间反转

力扣链接:反转链表 II

在这里插入图片描述

解决这道题目我们有两种方法,一种是头插法,一种是穿针引线法。

(1.1) 头插法

头插法反转的整体思想:在需要反转的区间里,每遍历到一个节点,让这个新节点来到反转部分的起始位置。

在这里插入图片描述

这个插入过程就是前面所学习的带虚拟结点的插入操作,每走一步都要考虑各种指针怎么指,既要将结点摘下来接到对应的位置上,还要保证后续结点能够找到。

public ListNode reverseBetween(ListNode head, int left, int right) {
    // 设置 dummyNode 是这一类问题的一般做法(统一节点不同位置处理)
    ListNode dummyNode = new ListNode(-1);
    dummyNode.next = head;
    ListNode pre = dummyNode;
    // 找到待反转区间的前一个节点
    for (int i = 0; i < left - 1; i++) {
        pre = pre.next;
    }
    // 指向反转区间的起始节点
    ListNode cur = pre.next;
    ListNode next;
    // 遍历区间内的节点并将其移向待反转区间的起始位置
    for (int i = 0; i < right - left; i++) {
        next = cur.next;
        cur.next = next.next;
        next.next = pre.next;
        pre.next = next;
    }
    return dummyNode.next;
}

(1.2) 穿针引线法

穿针引线法的整体思想:首先找到待反转区间的开始节点与结束节点以及反转区间的前一个节点以及后一个节点。将区间内节点进行反转,最后再按照我们记录的标识点进行连接即可。

在这里插入图片描述

public ListNode reverseBetween(ListNode head, int left, int right) {
    // 因为头节点有可能发生变化,使用虚拟头节点可以避免复杂的分类讨论
    ListNode dummyNode = new ListNode(-1);
    dummyNode.next = head;
    ListNode pre = dummyNode;
    // 第 1 步:从虚拟头节点走 left - 1 步,来到 left 节点的前一个节点
    for (int i = 0; i < left - 1; i++) {
        pre = pre.next;
    }
    // 第 2 步:从 pre 再走 right - left + 1 步,来到 right 节点
    ListNode rightNode = pre;
    for (int i = 0; i < right - left + 1; i++) {
        rightNode = rightNode.next;
    }
    // 第 3 步:切出一个子链表
    ListNode leftNode = pre.next;
    ListNode succ = rightNode.next;
    // 模拟设置为链表末尾节点指向null
    rightNode.next = null;

    // 第 4 步:反转链表的子区间
    reverseLinkedList(leftNode);
    // 第 5 步:接回到原来的链表中
    pre.next = rightNode;
    leftNode.next = succ;
    return dummyNode.next;
}
	// 反转链表
	private void reverseLinkedList(ListNode head) {
		ListNode pre = null;
		ListNode cur = head;
		while (cur != null) {
	    ListNode next = cur.next;
	    cur.next = pre;
	    pre = cur;
	    cur = next;
		}
	}

(2) 两两交换链表中的节点

力扣链接:两两交换链表中的节点

在这里插入图片描述

当链表存在两个及以上未遍历到的节点时,我们按照如下步骤模拟反转就行了。示意图如下:

在这里插入图片描述

    public ListNode swapPairs(ListNode head) {
        ListNode dummyNode = new ListNode();
        dummyNode.next = head;
        ListNode cur = dummyNode.next,pre = dummyNode;
        //在至少存在两个节点时进行翻转
        while(cur != null && cur.next != null){ 
            ListNode next = cur.next;
            cur.next = next.next;
            next.next = cur;
            pre.next = next;
            pre = cur;
            cur = cur.next;
        }
        return dummyNode.next;
    }

(3) 两数相加

力扣链接: 两数相加 II

在这里插入图片描述

我们可以用栈轻松解决就不在此赘述代码实现:先将两个链表的元素分别压栈,然后再一起出栈,将两个结果分别计算。之后对计算结果取模,模数保存到新的链表中,进位保存到下一轮。以此往复即可。

我们主要讲解如何针对链表操作,由于本题的链表从首到尾刚好是从最高位到最低位,而我们正常加法运算都是最低尾开始计算的,因此我们需要先将两条链表进行反转操作,然后再同时遍历每一位模拟加法运算即可轻松解决问题。

class Solution {
 public ListNode addTwoNumbers(ListNode l1, ListNode l2) {
     	// 定义虚拟头节点开始记录新链表
        ListNode ans = new ListNode();
     	// 反转两条链表
        ListNode rl1 = reverseListNode(l1);
        ListNode rl2 = reverseListNode(l2);
     	// 记录进位
        int add = 0;
     	// 链表不为空,进位不为0则持续计算
        while (rl1 != null || rl2 != null || add != 0) {
            int sum = 0;
            // 若不为空,则累加上该节点的值,同时后移一位
            if (rl1 != null) {
                sum += rl1.val;
                rl1 = rl1.next;
            }
            if (rl2 != null) {
                sum += rl2.val;
                rl2 = rl2.next;
            }
            // 加上上一轮的进位
            sum += add;
            // 记录本轮的进位
            add = sum / 10;
            // 计算进位后的数
            sum %= 10;
            // 从低位到高位拼接新的链表
            // ans -> 0 -> 7  8  ==>  ans -> 8 -> 0 -> 7  
            ListNode node = new ListNode(sum);
            node.next = ans.next;
            ans.next = node;
        }
        return ans.next;
    }
	// 直接操作反转链表
    public static ListNode reverseListNode(ListNode head) {
        ListNode pre = null;
        while (head != null) {
            ListNode next = head.next;
            head.next = pre;
            pre = head;
            head = next;
        }
        return pre;
    }
}

(4) 回文链表

力扣链接:回文链表

在这里插入图片描述

在上一篇中我们讲解了判断回文链表可以使用集合+双指针或者栈来解决,我们还可以通过将双指针+反转一半链表再进行比较判断来实现。

class Solution {
    public boolean isPalindrome(ListNode head) {
        if (head == null) {
            return true;
        }

        // 找到前半部分链表的尾节点并反转后半部分链表
        ListNode firstHalfEnd = endOfFirstHalf(head);
        ListNode secondHalfStart = reverseList(firstHalfEnd.next);

        // 判断是否回文
        ListNode p1 = head;
        ListNode p2 = secondHalfStart;
        boolean result = true;
        while (result && p2 != null) {
            if (p1.val != p2.val) {
                result = false;
            }
            p1 = p1.next;
            p2 = p2.next;
        }        

        // 还原链表并返回结果
        firstHalfEnd.next = reverseList(secondHalfStart);
        return result;
    }
	// 反转链表
    private ListNode reverseList(ListNode head) {
        ListNode prev = null;
        ListNode curr = head;
        while (curr != null) {
            ListNode nextTemp = curr.next;
            curr.next = prev;
            prev = curr;
            curr = nextTemp;
        }
        return prev;
    }
	// 通过双指针找到后半部分第一个节点
    private ListNode endOfFirstHalf(ListNode head) {
        ListNode fast = head;
        ListNode slow = head;
        while (fast.next != null && fast.next.next != null) {
            fast = fast.next.next;
            slow = slow.next;
        }
        return slow;
    }
}

(5) K 个一组翻转链表

力扣链接:K 个一组翻转链表

在这里插入图片描述

这一题其实可以看做是指定区间反转的进阶形式,只要我们足够细心,其实这一题并不难。我们同样要使用到穿针引线法来解决问题,相当于使用穿针引线法反转若干个指定区间。

    public ListNode reverseKGroup(ListNode head, int k) {
       ListNode dummyNode = new ListNode();
        dummyNode.next = head;
        // 记录穿针引线法需要使用的四个标记点
        ListNode pre = dummyNode, right, left, succ;
        // 反转若干个指定区间
        while (true) {
            right = pre;
            // 1.寻找到反转区间right节点
            for (int i = 0; i < k; i++) {
                // 当一组不足k个时,表明不需要反转了,直接返回结果
                if (right.next == null) {
                    return dummyNode.next;
                }
                right = right.next;
            }
            // 2.记录left节点以及succ节点
            left = pre.next;
            succ = right.next;
            // 3.反转指定区间
            right.next = null;
            right = reverse(left);
            // 4.穿针引线拼接链表
            pre.next = right;
            left.next = succ;
            // 5.更新为下一个区间的起始节点的前一个节点
            pre = left;
        }
    }
	// 反转链表
    public static ListNode reverse(ListNode head) {
        ListNode pre = null;
        while (head != null) {
            ListNode next = head.next;
            head.next = pre;
            pre = head;
            head = next;
        }
        return pre;
    }

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

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

相关文章

代码随想录算法训练营第45天| 139.单词拆分 多重背包

JAVA代码编写 139.单词拆分 给你一个字符串 s 和一个字符串列表 wordDict 作为字典。请你判断是否可以利用字典中出现的单词拼接出 s 。 **注意&#xff1a;**不要求字典中出现的单词全部都使用&#xff0c;并且字典中的单词可以重复使用。 示例 1&#xff1a; 输入: s &…

如何利用人工智能+物联网技术实现自动化设备生产

随着科技的发展与行业竞争的日益激烈&#xff0c;制造业也逐渐走向智能化发展。制造业的改革是利用物联网技术和自动化设备&#xff0c;实现生产线的智能化和自适应生产&#xff0c;优化生产流程&#xff0c;提高生产效率和质量&#xff0c;为企业创造更大的价值。 方案概述 智…

Linux 压缩、文件传输与安装

目录 1. 压缩 1.1 tar 1.2 gzip 1.3 zip 1.4 rar 2 文件传输 2.1 网站下载 2.2 scp 传输 2.3 rz 和 sz 2.4 xftp 3.安装 3.1 编译安装 &#xff08;ngnix&#xff09; 3.2 rpm 安装 3.3 yum 安装 1. 压缩 1.1 tar 使用 tar 压缩文件时&#xff0c;会保留源文件…

使用MetaMask + Ganache搭建本地私有网络并实现合约部署与互动

我使用Remix编写合约&#xff0c;MetaMask钱包工具和Ganache搭建了一个私有网络&#xff0c;并且实现了合约的部署和互动。 在前面的博客中提到了 Remix在线环境及钱包申请 以及 Solidity的基本语法 &#xff0c;没看过的小伙伴可以点击链接查看一下&#xff0c;都是在本专栏下…

智能时代:互联网+如何改变我们的生活与工作

引言 随着科技的不断进步和互联网的普及&#xff0c;我们正处在一个智能时代。这个时代被互联网所定义&#xff0c;它深刻地改变了我们的生活和工作方式。从社交互动到日常工作&#xff0c;智能时代的影响无处不在&#xff0c;给人们带来了前所未有的变革和机遇。 互联网的涌…

模电·放大电路的分析方法——图解法

放大电路的分析方法——图解法 静态工作点的分析电压放大倍数的分析波形非线性失真的分析直流负载线与交流负载线图解法的适用范围 在实际测出放大管的输入特性、输出特性和已知放大电路中其它各元件参数的情况下&#xff0c;利用作图的方法对放大电路进行分析即为图解法。 静…

FacetWP WordPress网站高级筛选过滤插件(含所有扩展)

点击阅读FacetWP WordPress网站高级筛选过滤插件原文 FacetWP WordPress网站高级筛选过滤插件向电子商务网站、资源库、搜索页面等添加分面搜索。FacetWP 的过滤元素&#xff08;称为 facets&#xff09;动态调整以适应用户输入。这有助于防止出现“未找到结果”&#xff0c;从…

OpenSSL 编程指南

目录 前言初始化SSL库创建SSL 上下文接口(SSL_CTX)安装证书和私钥加载证书(客户端/服务端证书)加载私钥/公钥加载CA证书设置对端证书验证例1 SSL服务端安装证书例2 客户端安装证书创建和安装SSL结构建立TCP/IP连接客户端创建socket服务端创建连接创建SSL结构中的BIOSSL握手服务…

【MATLAB源码-第99期】基于matlab的OFDM系统卡尔曼滤波(kalman)信道估计,对比LS,MMSE。

操作环境&#xff1a; MATLAB 2022a 1、算法描述 卡尔曼滤波器&#xff08;Kalman Filter&#xff09;是一种有效的递归滤波器&#xff0c;它能够从一系列的含有噪声的测量中估计动态系统的状态。在无线通信领域&#xff0c;尤其是在正交频分复用&#xff08;OFDM&#xff0…

Pandas中的Series(第1讲)

Pandas中的Series(第1讲)         🍹博主 侯小啾 感谢您的支持与信赖。☀️ 🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔…

Redis基础入门

第1章&#xff1a;引言 大家好&#xff01;我是小黑&#xff0c;今天咱们来聊聊Redis。Redis&#xff0c;这个名字你可能在不少地方听过&#xff0c;尤其是在后端开发领域&#xff0c;它可是个大名鼎鼎的角色。&#xff0c;Redis是一个开源的内存中数据结构存储系统&#xff0…

利用jdbc对数据库进行增删改查

步骤/过程&#xff1a; 1&#xff0c;导入驱动包 2&#xff0c;加载驱动包 3&#xff0c;输入信息&#xff0c;进行数据库连接 4&#xff0c;创建 statement对象 5&#xff0c;执行sql语句 6&#xff0c;如果是查询操作&#xff0c;利用ResultSet处理数据&#xf…

Plantuml之类图语法介绍(十六)

简介&#xff1a; CSDN博客专家&#xff0c;专注Android/Linux系统&#xff0c;分享多mic语音方案、音视频、编解码等技术&#xff0c;与大家一起成长&#xff01; 优质专栏&#xff1a;Audio工程师进阶系列【原创干货持续更新中……】&#x1f680; 优质专栏&#xff1a;多媒…

Ubuntu中编译出Windows的可执行程序(.exe)

1、前言 在嵌入式开发中&#xff0c;交叉编译是很常见的情况&#xff0c;如果你把Windows电脑也看做一块高性能的开发板&#xff0c;那在Ubuntu中编译出Windows上运行的可执行程序也是很好理解的行为。 2、安装mingw64环境 sudo apt-get install mingw-w64 3、测试编译链是否安…

淘宝权益玩法平台的Serverless化实践

通过对权益玩法平台现有业务应用的Serverless化改造&#xff0c;权益团队在双十一期间完美地支撑了业务需求&#xff0c;在研发效率、运维保障等方面都体现出了很高的价值和收益。 项目背景 淘宝权益平台是负责淘宝权益营销的核心团队&#xff0c;团队除了负责拉菲权益平台外&a…

反汇编语言区分函数和运算符

在汇编语言中&#xff0c;函数和运算符可以通过一些特定的指令和约定来区分。 函数&#xff1a; 函数通常由一系列指令组成&#xff0c;用于执行特定的任务或操作。函数通常具有入口点和出口点&#xff0c;分别表示函数的开始和结束位置。函数通常包含参数传递、局部变量的分配…

VIM光标移动和翻页快捷键-包含vim帮助文档截图

光标移动到行首(行首没有空格)&#xff1a; ^ 光标移动到行首(行首有空格)&#xff1a; 数字0 光标移动到行尾&#xff1a; $ 移动到指定行&#xff1a;7G(数字加一个大G&#xff09; 光标移动到文件开始&#xff1a;gg(两个小g) 光标移动到文件末尾&#xff1a;G(一个大G&…

根据对数器找规律、根据数据量猜题目解法

题目一 小虎去买苹果&#xff0c;商店只提供两种类型的塑料袋&#xff0c;每种类型都有任意数量。1&#xff09;能装下6个苹果的袋子2&#xff09;能装下8个苹果的袋子小虎可以自由使用两种袋子来装苹果&#xff0c;但是小虎有强迫症&#xff0c;他要求自己使用的袋子数量必须…

Weblogic T3协议反序列化漏洞

文章目录 1. Weblogic T3协议反序列化漏洞1.1 漏洞描述1.2 基本原理1.3 漏洞复现1.4 修复建议 1. Weblogic T3协议反序列化漏洞 1.1 漏洞描述 说明内容漏洞编号CVE-2018-2628漏洞名称Weblogic T3协议反序列化漏洞漏洞评级高危影响范围Weblogic 10.3.6.0Weblogic 12.1.3.0Webl…

2024年甘肃省职业院校技能大赛信息安全管理与评估赛项一阶段样题一

2024年甘肃省职业院校技能大赛高职学生组电子与信息大类信息安全管理与评估赛项样题一 竞赛需要完成三个阶段的任务&#xff0c;分别完成三个模块&#xff0c;总分共计 1000分。三个模块内容和分值分别是&#xff1a; 1.第一阶段&#xff1a;模块一 网络平台搭建与设备安全防…