LeetCode 718 - 最长重复子数组

news2025/3/4 5:30:00

LeetCode 718 - 最长重复子数组 是一个典型的数组和字符串问题,适合考察动态规划、滑动窗口和二分查找等多种编程能力。掌握其多种解法及变体能够有效提高处理字符串和数组算法的能力。


题目描述

  • 输入: 两个整数数组 nums1nums2
  • 输出: 两个数组中存在的最长的连续重复子数组的长度。
  • 要求: 时间复杂度尽可能优化。

示例

输入: nums1 = [1,2,3,2,1], nums2 = [3,2,1,4,7]
输出: 3
解释: 最长的重复子数组是 [3,2,1], 长度为 3。

解法分析及实现

解法 1:二维动态规划

思路
  • 定义 dp[i][j] 表示:
    • 以下标 i-1(对于 nums1)结尾的子数组和以下标 j-1(对于 nums2)结尾的子数组的最长公共子数组长度。
  • 状态转移方程:
    • 如果 nums1[i-1] == nums2[j-1]:
      dp[i][j] = dp[i-1][j-1] + 1
      
    • 否则:
      dp[i][j] = 0
      
  • 初始化:
    • dp[i][0] = 0 (空序列和任何序列无公共子数组)。
    • dp[0][j] = 0 (空序列和任何序列无公共子数组)。
  • 最终答案:
    • 遍历所有的 dp[i][j],取最大值。
模板代码
class Solution {
    public int findLength(int[] nums1, int[] nums2) {
        int m = nums1.length, n = nums2.length;
        int[][] dp = new int[m + 1][n + 1];
        int maxLength = 0;

        // 动态规划填表
        for (int i = 1; i <= m; i++) {
            for (int j = 1; j <= n; j++) {
                if (nums1[i - 1] == nums2[j - 1]) {
                    dp[i][j] = dp[i - 1][j - 1] + 1;
                    maxLength = Math.max(maxLength, dp[i][j]);
                }
            }
        }
        return maxLength;
    }
}
复杂度分析
  • 时间复杂度: O(m * n),其中 mn 是数组长度。
  • 空间复杂度: O(m * n),由于用了二维 DP 表保存状态。

解法 2:滚动数组优化动态规划(空间优化)

优化思路
  • 注意到在计算 dp[i][j] 时,只需要用到 dp[i-1][j-1] 的值。因此,可以用一维数组代替二维数组,进一步优化空间。
  • 滚动更新:从右往左遍历 nums2,避免覆盖之前的状态。
模板代码
class Solution {
    public int findLength(int[] nums1, int[] nums2) {
        int m = nums1.length, n = nums2.length;
        int[] dp = new int[n + 1];
        int maxLength = 0;

        // 动态规划填表
        for (int i = 1; i <= m; i++) {
            for (int j = n; j >= 1; j--) { // 注意,从右往左遍历 nums2
                if (nums1[i - 1] == nums2[j - 1]) {
                    dp[j] = dp[j - 1] + 1;
                    maxLength = Math.max(maxLength, dp[j]);
                } else {
                    dp[j] = 0; // 无法连续时,长度清零
                }
            }
        }
        return maxLength;
    }
}
复杂度分析
  • 时间复杂度: O(m * n),遍历数组。
  • 空间复杂度: O(n),使用了一维数组。

解法 3:滑动窗口法

核心思想
  • 假设固定一个数组 nums1,滑动另一个数组 nums2 进行比较。
  • 在滑动的过程中寻找最大公共子数组长度。
    • 如果 nums1nums2 不完全对齐,就逐步滑动 nums2,逐一比较。
模板代码
class Solution {
    private int maxLen(int[] A, int[] B, int offsetA, int offsetB, int length) {
        int maxLen = 0, currentLen = 0;
        for (int i = 0; i < length; i++) {
            if (A[offsetA + i] == B[offsetB + i]) {
                currentLen++;
                maxLen = Math.max(maxLen, currentLen);
            } else {
                currentLen = 0;
            }
        }
        return maxLen;
    }

    public int findLength(int[] nums1, int[] nums2) {
        int m = nums1.length, n = nums2.length;
        int maxLength = 0;

        // 滑动 nums1 相对于 nums2
        for (int i = 0; i < m; i++) {
            int len = Math.min(n, m - i); // 两数组能对齐的长度
            maxLength = Math.max(maxLength, maxLen(nums1, nums2, i, 0, len));
        }
        // 滑动 nums2 相对于 nums1
        for (int j = 0; j < n; j++) {
            int len = Math.min(m, n - j); // 两数组能对齐的长度
            maxLength = Math.max(maxLength, maxLen(nums1, nums2, 0, j, len));
        }
        return maxLength;
    }
}
复杂度分析
  • 时间复杂度: O((m + n) * min(m, n)),滑动两遍数组。
  • 空间复杂度: O(1),原地比较,不需要额外空间。

解法 4:二分查找 + 哈希

核心思想
  • 使用滑动窗口与哈希表结合,通过 二分查找 在所有可能的子数组长度中快速找到最长重复子数组长度。
    1. 利用暴力检查或 Rabin-Karp 哈希验证是否存在长度为 mid 的公共子数组:
      • 用窗口提取长度为 mid 的子数组,进行比对。
    2. 使用二分查找:
      • 如果某个长度是可行的(存在共同子数组),增加长度;
      • 如果不可行,减小长度。
模板代码
import java.util.HashSet;

class Solution {
    public int findLength(int[] nums1, int[] nums2) {
        int left = 0, right = Math.min(nums1.length, nums2.length);
        int result = 0;

        while (left <= right) {
            int mid = left + (right - left) / 2;
            if (check(nums1, nums2, mid)) {
                result = mid;
                left = mid + 1; // 尝试更长的长度
            } else {
                right = mid - 1; // 尝试更短的长度
            }
        }
        return result;
    }

    private boolean check(int[] nums1, int[] nums2, int len) {
        // 使用 HashSet 存储 nums1 长度为 len 的子数组
        HashSet<String> seen = new HashSet<>();
        for (int i = 0; i + len <= nums1.length; i++) {
            seen.add(arrayToString(nums1, i, len));
        }
        // 检查 nums2 是否包含相同子数组
        for (int j = 0; j + len <= nums2.length; j++) {
            if (seen.contains(arrayToString(nums2, j, len))) {
                return true;
            }
        }
        return false;
    }

    private String arrayToString(int[] nums, int start, int len) {
        StringBuilder sb = new StringBuilder();
        for (int i = start; i < start + len; i++) {
            sb.append(nums[i]).append(",");
        }
        return sb.toString();
    }
}
复杂度分析
  • 时间复杂度: O(log(min(m, n)) * (m + n)),二分查找次数乘以每次滑动窗口检查的时间。
  • 空间复杂度: O(min(m, n)),为哈希表的空间。

变体问题

  1. 最长公共子序列 (LCS)

    • 两数组中非连续元素符合顺序的最长子序列长度。
    • DP 思路与本题类似,但不要求元素连续。
  2. 最长公共前缀 (Longest Common Prefix)

    • 但不局限于以何种顺序,重点是找最大前缀。
  3. 编辑距离问题

    • 在两个字符串之间,进行最少字符操作(包括插入、删除、替换)使其相等。
  4. 字符串匹配问题(KMP 或 Rabin Karp)

    • 查找两个字符串的最大匹配区段。

快速 AC 总结

  1. 明确问题是连续子数组,直接优先使用滚动 DP 模板。
  2. 熟悉时间复杂度要求:若需要 O(m * n) 解决,选择滑动窗口或动态规划。
  3. 若需寻求更高效方法,尝试二分和哈希结合的解法。
  4. 练习经典变体问题如 LCS,使技术迁移更灵活。

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

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

相关文章

[Windows] 免费电脑控制手机软件 极限投屏_正式版_3.0.1 (QtScrcpy作者开发)

[Windows] 极限投屏_正式版 链接&#xff1a;https://pan.xunlei.com/s/VOKJf8Z1u5z-cHcTsRpSd89tA1?pwdu5ub# 新增功能(Future)&#xff1a; 支持安卓14(Supports Android 14)提高投屏成功率(Improve the success rate of mirror)加快投屏速度(Accelerate screen mirrorin…

C++初阶—list类

第一章&#xff1a;list的介绍及使用 1.1 list的介绍 list是可以在常数范围内在任意位置进行插入和删除的序列式容器&#xff0c;并且该容器可以前后双向迭代。list的底层是双向链表结构&#xff0c;双向链表中每个元素存储在互不相关的独立节点中&#xff0c;在节点中通过指…

【子网掩码计算器:Python + Tkinter 实现】

子网掩码计算器&#xff1a;Python Tkinter 实现 引言代码功能概述代码实现思路1. 界面设计2. 功能实现3. 事件处理 子网掩码计算器实现步骤1. 导入必要的库2. 定义主窗口类 SubnetCalculatorApp3. 创建菜单栏4. 创建界面组件5. 判断 IP 地址类别6. 计算子网信息7. 其他功能函…

入门基础项目(SpringBoot+Vue)

文章目录 1. css布局相关2. JS3. Vue 脚手架搭建4. ElementUI4.1 引入ElementUI4.2 首页4.2.1 整体框架4.2.2 Aside-logo4.2.3 Aside-菜单4.2.4 Header-左侧4.2.5 Header-右侧4.2.6 iconfont 自定义图标4.2.7 完整代码 4.3 封装前后端交互工具 axios4.3.1 安装 axios4.3.2 /src…

车载以太网-基于linux的ICMP协议

对于车载以太网-ICMP的技术要求: /** ICMP报文格式解析* -----------------* ICMP协议用于网络诊断和错误报告,常见应用包括Ping测试。* ICMP报文结构包括:IP头部、ICMP头部和ICMP数据部分。* 下面详细介绍每个部分的结构、字段的作用以及如何解析它们。* * ICMP头部结构:*…

虚拟机快照与linux的目录结构

虚拟机快照是对虚拟机某一时刻状态的完整捕获&#xff0c;包括内存、磁盘、配置及虚拟硬件状态等&#xff0c;保存为独立文件。 其作用主要有数据备份恢复、方便系统测试实验、用于灾难恢复以及数据对比分析。具有快速创建和恢复、占用空间小、可多个快照并存的特点。在管理维…

Unity小功能实现:鼠标点击移动物体

1、功能描述 当玩家点击鼠标时&#xff0c;场景中的物体会移动到鼠标点击的位置。这个功能可以用于控制角色移动、放置物体等场景。 2、实现步骤 创建Unity项目&#xff1a;首先&#xff0c;打开Unity并创建一个新的3D项目。 添加3D物体&#xff1a;在场景中创建一个3D物体&am…

算法题笔记(自用)——Python

目录 一. 进制&位运算&ASCAII 二. format格式化输出 1. 基本用法 2. 位置参数 3. 格式化数字 4. 对齐和填充 5. 格式化二进制、八进制、十六进制 6. 格式化百分比 7. 格式化科学计数法 8. 格式化字符串字面量&#xff08;f-string&#xff09; 三. 字符串 使…

爬虫系列之【数据解析之JSON】《三》

目录 前置知识 一、 json.loads()&#xff1a;JSON 转 Python 数据 二、json.dump()&#xff1a;python数据 转 json 并写入文件 三、json.loads() &#xff1a;json 转 python数据 四、json.load() &#xff1a;json 转 python数据&#xff08;在文件操作中更方便&#xf…

了解Java集合的概念和体系:Collection<T>、Collections与Stream的使用

学习目标 本文知识是对集合层级的介绍&#xff0c;应用开发中实际使用的是他们的子级&#xff0c;感兴趣的小伙伴或者想深入了解有关Java集合知识的朋友可以选择阅读&#xff01; Stream的方法使用使用部分代码块内大多有两种实现方式&#xff0c;是为了更好的理解方法底层的代…

进程优先级和进程切换 ─── linux第12课

目录 Z(zombie)-僵尸进程 僵尸进程危害 孤儿进程 ​编辑 进程优先级 查看系统进程 用top命令更改已存在进程的nice&#xff1a; 其他概念 进程切换 进程如何切换 进程的调度 进程 内核数据结构 代码和数据 创建进程时 ,先创建内核数据结构 再加载代码和数据 进程退…

光速解决phpstudy无法启动MySQL服务

问题描述 在初步安装使用phpstudy时&#xff0c;会出现mysql服务启动失败的情况&#xff0c;具体表现为点击一键启动后&#xff0c;mysql启动又立即停止 原因及解决方法&#xff1a; phpstudy数据库无法启动有以下几个原因&#xff0c;可以看看自己是第几种&#xff1a; 服务名…

txt 转 json 使用python语言

需求: 把如下的txt文档转成json输出 代码 import jsondef txt_to_json(input_file, output_file):data_list []with open(input_file, r, encodingutf-8) as f:for line in f:# 分割数据并去除换行符parts line.strip().split(,)print(f"{parts}")print(type(par…

Cargo, the Rust package manager, is not installed or is not on PATH.

今天在Windows操作系统上通过pip 安装jupyter的时候遇到这个报错&#xff0c;Cargo, the Rust package manager, is not installed or is not on PATH.。 解决办法 官网&#xff1a;https://rustup.rs/# 下载&#xff1a;https://win.rustup.rs/x86_64 安装完成之后&#xff0c…

JavaWeb个人笔记

技术栈 前端 : HTML CSS JavaScript ES6 Nodejs npm vite vue3 router pinia axios element-plus 后端&#xff1a;HTTP xml Tomcat Servlet Request Response Cookie Sesssion Filter Listener MySQL JDBC Druid Jackson lombok jwt . HTML <!DOCTYPE html> 文档声…

http报文的content-type参数和spring mvc传参问题

很早之前博主聊过HTTP的报文结构以及其中和传参相关的重要参数content-type还有spring mvc&#xff0c;以前的三篇文章&#xff1a; HTTP与HTTPS协议详解&#xff1a;基础与安全机制-CSDN博客 详解Http的Content-Type_content-type application-CSDN博客 如何在Spring Boot中…

14. LangChain项目实战1——基于公司制度RAG回答机器人

教学视频&#xff1a; 12. 基于Gradio搭建基于公司制度RAG_哔哩哔哩_bilibilihttps://www.bilibili.com/video/BV11VXRYTErZ/ 环境配置&#xff1a; python版本&#xff1a;3.10.8 服务器&#xff1a;Ubuntu 依赖包requirements.txt文件内容&#xff1a; aiofiles23.2.1 …

FPGA开发,使用Deepseek V3还是R1(6):以滤波器为例

以下都是Deepseek生成的答案 FPGA开发&#xff0c;使用Deepseek V3还是R1&#xff08;1&#xff09;&#xff1a;应用场景 FPGA开发&#xff0c;使用Deepseek V3还是R1&#xff08;2&#xff09;&#xff1a;V3和R1的区别 FPGA开发&#xff0c;使用Deepseek V3还是R1&#x…

JVM常用概念之垃圾回收设计与停顿

在我们应用程序运行期间&#xff0c;我们是需要尽可能避免垃圾回收。 图1&#xff1a;不同垃圾回收器的设计&#xff08;黄色代表STW&#xff0c;绿色代表并发&#xff09; 实验 计算机配置 Hardware Overview:Model Name: MacBook ProModel Identifier: MacBookPro14,2Pro…

uniapp-原生android插件开发摘要

uni-app在App侧的原生扩展插件&#xff0c;支持使用java、object-c等原生语言编写&#xff0c;从HBuilderX 3.6起&#xff0c;新增支持了使用uts来开发原生插件。 基础项目 UniPlugin-Hello-AS工程请在App离线SDK中查找 基础项目(App离线SDK)已经配置好了自定义插件所需要的…