arry-718 Maximum Length of Repeated Subarray

题目:Input:
A: [1,2,3,2,1]
B: [3,2,1,4,7]
Output: 3
Explanation:
The repeated subarray with maximum length is [3, 2, 1].

思路:要找到两个数组中重复数据最长的子数组的长度。暴力枚举:每个A的下标i,分别与B的每个下标j作为起点,判断最长的重复数组长度。就像是移动两个指针。
例如i=0,j分别为0,1,2,3,4的时候,重复数组的长度分别为0,0,1,0,0。
当i=2,j=0的时候,重复数组的长度是3。接着还要计算i=2,j=1。如此一直计算到最后。这里的一个问题是当计算i=2,j=0找数组找到i=4,j=2结束之后,还有必要查找i=2,j=1位起始节点的子数组吗?有必要,在连续重复的数组中。例如数组A=[X,X,0,0,0,1],数组B=[0,0,0,0,1],可以看到以i=2,j=0,为起点,重复数组长度为3,;以i=2,j=1位起点,重复数组长度为4。所以这个暴力枚举没有可以省略操作的地方。
时间复杂度:A中每个元素都要和B中每个元素比较一次,O(m*n)

    public int findLength(int[] A, int[] B) {int len1 = A.length;int len2 = B.length;int maxLen = 0;for(int i=0;i<len1;i++){int j = 0;while(j<len2){if(A[i] == B[j]){int count = 0;int idxA = i;int idxB = j;while(idxA<len1 && idxB <len2 && A[idxA++] == B[idxB++]) count++;maxLen = Math.max(maxLen,count);}j++;}}return maxLen;}

学习:动态规划。上面分析是以数组的起点开始思考,动态规划以i,j作为子数组的终点元素,考虑从(0,0)到(i,j)中,重复数组的最长长度。
分析递归方程式。还以题目中的A、B数组为例。假设i=3,j=1,A[3]=B[1],那么dp[3][1] = dp[2][0]+1。假设i=3,j=4,A[3]B[4]A[3]≠B[4],那么dp[3][4]=dp[2][3]。得出dp[i][j] = dp[i-1][j-1]+1 如果A[i]=B[j]。这里之所以dp[i][j]与dp[i-1][j-1]有关系,而不是跟dp[i][j-1]或者dp[i-1][j]有关系,是因为这是求重复数组的长度。而求重复数组肯定是两个数组的下标同时移动。
分析初始化条件。如果i=0,如果A[i]=B[j]则dp[0][j]=1否则为0。如果j=0,如果A[i]=B[j]则dp[i][0]=1否则为0。

    public int findLengthV2(int[] A, int[] B) {if (A == null || B == null)return 0;int m = A.length;int n = B.length;int[][] dp = new int[m][n];int maxLen = 0;for (int i = 0; i < m; i++) {for (int j = 0; j < n; j++) {if (j == 0 || i == 0) {dp[i][j] = (A[i] == B[j] ? 1 : 0);} else if (A[i] == B[j]) {dp[i][j] = 1 + dp[i - 1][j - 1];maxLen = Math.max(maxLen, dp[i][j]);}}}return maxLen;}

思考:让我不能理解的地方是都是两层for循环,时间复杂度都是O(m*n)。方法二就比方法一快。相比较之下方式一多出了

while(idxA<len1 && idxB <len2 && A[idxA++] == B[idxB++]) count++;

这个步骤。方法二中只是简单的加法操作,当然更快。这样说来,方法一的时间复杂度就不是O(m*n)。当数组中重复元素多的情况下,两者速度相差就更大了。例如以下数组:
[59,59,59,93,59,59,59,59,59,59,59,59,59,59,59,59…]
[59,59,59,91,59,59,59,59,59,59,59,59,59,59,59,59..]
还有让我感觉神奇的地方是如果以(i,j)为子数组起点考虑问题,解决方法时间长;而以(i,j)为子数组终点考虑问题,竟然速度就快多了。这就是换个思路?再看看下面这段代码。

public int findLength(int[] a, int[] b) {int m = a.length, n = b.length;if (m == 0 || n == 0) return 0;int[][] dp = new int[m + 1][n + 1];int max = 0;for (int i = m - 1; i >= 0; i--)for (int j = n - 1; j >= 0; j--)max = Math.max(max, dp[i][j] = a[i] == b[j] ? 1 + dp[i + 1][j + 1] : 0);return max;        }

这个思路是以(i,j)作为子数组的开始节点,dp[i][j]存储以(i,j)为子数组起始节点的重复数组最长长度。这时候从最大的下标开始考虑。为什么从最大下标开始考虑,看下面的分析。
数组:A: [1,2,3,2,1] B: [3,2,1,4,7]
例如i=4,j=4,因为A[4]不等于B[4],所以dp[4][4]=0。
例如i=3,j=1,则dp[3][1]=dp[4][2]+1,应该是在以j=4,j=2这个数组重复数组最长长度+1,因为A[3]=B[1]。dp[3][1] 与dp[2][0]的关系就不是很确定。我可能因为向后移动了位置,重复数组变长,也可能变短。
开辟空间dp[m+1][n+1] ,而不是dp[m][n]是为了编码方便。
所以:暴力枚举与动态规划是两种思考问题的方式。动态规划在找递推关系的时候不是一定要有dp[i][j]与dp[i-1][j-1],也可能是dp[i+1][j+1]。根据题目的确定性关系确定递推式。

参考资料:
网页1
网页2

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

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

相关文章

第六十六期:软件架构之道的一次感悟

张泰峰 6月3日 写在前面 2019悄悄溜走一半&#xff0c;无论是离别的忧愁&#xff0c;还是成长路途的艰辛&#xff0c;都在心中滚烫。 距离上一篇文章已经很久了... 懒惰的博主不能将这一切归结于我的时间、我的规划、我的工作&#xff0c;只能怪自己懒......正所谓学如逆水行…

[Leetcode][第332题][JAVA][重新安排行程][欧拉回路 / 欧拉通路][优先队列][DFS]

【问题描述】[中等] 【解答思路】 递归 复杂度 class Solution {Map<String, PriorityQueue<String>> map new HashMap<String, PriorityQueue<String>>();List<String> itinerary new LinkedList<String>();public List<String&g…

不会配置HTTPS?给我5分钟,手把手教你

本文针对不会配置HTTPS或者小白开发着&#xff0c;请大佬们自动忽略。非广告&#xff0c;心得分享&#xff0c;勿喷&#xff0c;谢谢。 正文开始 01、关于 FreeSSL.cnFreeSSL.cn 是一个免费提供 HTTPS 证书申请、HTTPS 证书管理和 HTTPS 证书到期提醒服务的网站&#xff0c;旨在…

第三十三期:使用wireshark抓包分析-抓包实用技巧

杰哥很忙 7月9日 前言 本文整理一下日常抓包使用的一些方法及抓包分析的一些方法。 本文基于wireshark2.2.6版本进行抓包处理。其他版本使用方式大同小异。 自定义捕获条件 wireshark可以将抓包数据保存到硬盘上。若需要长时间抓包的话&#xff0c;需要防止内存过大&#x…

Floyd判圈算法

leetcode习题287 Find the Duplicate Number 在答案中看到了floyd’s tortoise and hare 算法&#xff0c;知道了如果有限状态机、迭代函数或者链表存在环&#xff0c;那么是需要算法检测环是否存在。检测算法有三种:Floyd龟兔算法、Brent算法、Gosper算法。 Floyd龟兔算法 算…

【数据结构与算法】【算法思想】【联系与区别】回溯 贪心 动态规划 分治

4种算法思想比较与联系 如果将贪心&#xff0c;分治&#xff0c;回溯和动态规划四种算法思想分类&#xff0c;那贪心&#xff0c;回溯&#xff0c;动态规划可归为一类&#xff0c;而分治单独可以作为一类&#xff0c;因为它跟其他是三个都不大一样。 因为前三个算法解决问题的…

Idea debug时报错:Command line is too long

问题&#xff1a;git pull下的项目&#xff0c;debug时&#xff0c;报错如下图 解决方法 在项目/.idea/workspace.xml文件中添加一行代码如下 <component name"PropertiesComponent">...<property name"dynamic.classpath" value"true"…

第三十四期:一次非常有意思的sql优化经历

风过无痕的博客 6月24日 场景 我用的数据库是mysql5.6&#xff0c;下面简单的介绍下场景 课程表&#xff1a; create table Course(c_id int PRIMARY KEY,name varchar(10)) 数据100条 学生表: create table Student(id int PRIMARY KEY,name varchar(10)) 数据70000条…

array专题6

78. Subsets 思路1&#xff1a;深度优先搜索&#xff1a;每个位置都有选&#xff0c;和 不选两种选择。退出条件是下标出界。 public List<List<Integer>> subsetsV3(int[] nums) {List<List<Integer>> result new ArrayList<List<Integer>…

[Leetcode][第657题][JAVA][机器人能否返回原点][数组][HashMap]

【问题描述】[简单] 【解答思路】 遍历方向 看是否回到原点 或者 “上下” “左右”两个方向的数量是否相等 1. 方向 时间复杂度&#xff1a;O(N) 空间复杂度&#xff1a;O(1) class Solution {public boolean judgeCircle(String moves) {int x 0,y 0;int len moves.len…

网站公共部分的复用

一个网站的公共部分因为是一样的&#xff0c;可以通过公共部分的复用来减少代码的重复&#xff0c;同时也利于代码的后期维护。 关于公共部分复用的方法有很多&#xff0c;这里提供一些方法&#xff08;持续更新&#xff09; 利用ajax请求组件&#xff08;单个HTML&#xff09;…

第三十五期:当我们在讨论CQRS时,我们在讨论些神马?

thz 6月18日 当我写下这个标题的时候&#xff0c;我就有些后悔了&#xff0c;题目有点大&#xff0c;不太好控制。但我还是打算尝试一下&#xff0c;通过这篇内容来说清楚CQRS模式&#xff0c;以及和这个模式关联的其它东西。希望我能说得清楚&#xff0c;你能看得明白&#x…

【数据结构与算法】【算法思想】回溯算法

贪心算法 回溯算法 分治算法 动态规划 回溯算法思想应用广泛&#xff0c;除了用来指导深度优先搜索这种经典算法设计之外&#xff0c;还可以用在如正则表达式匹配&#xff0c;编译原理中的语法分析等。 除此之外&#xff0c;很多经典的数学问题都可以用回溯算法解决&#xff…

90 Subsets II

90 Subsets II leetcode第90题&#xff0c;用深度优先搜索的思路写出了三种不同的代码。一道题目&#xff0c;同样是深度优先搜索的问题&#xff0c;因为看问题角度不同&#xff0c;思路不同&#xff0c;代码也不一样 /*** [1,2,2,3] [] [1] [1,2] [1,2](重复) 情况&#xff1a…

2019 DDCTF 部分writeup

网上的wp已经很多了&#xff0c;但wp普遍很简略。我尽量写的详细一点。 一、WEB 滴~ 拿到题目后首先右键查看源代码&#xff0c;发现图片是以base64传送的 而且看url发现里面应该是包含了文件名&#xff0c;并且用了某个编码。测试过后是转16进制ascii码后两层bases64 &#xf…

第三十六期:如果把线程当作一个人来对待,所有问题都瞬间明白了

7月8日 以下文章来源于编程新说 &#xff0c;作者编程新说李新杰 多线程的问题都曾经困扰过每个开发人员&#xff0c;今天将从全新视角来解说&#xff0c;希望读者都能明白。 强烈建议去运行下文章中的示例代码&#xff0c;自己体会下。 问题究竟出在哪里&#xff1f; 一个…

[Leetcode][第322题][JAVA][零钱兑换][回溯][记忆化搜索][动态规划]

【问题描述】[中等] 【解答思路】 1. 递归&#xff08;超时&#xff09; class Solution {int res Integer.MAX_VALUE;public int coinChange(int[] coins, int amount) {if(coins.length 0){return -1;}findWay(coins,amount,0);// 如果没有任何一种硬币组合能组成总金额&…

用BST解决729. My Calendar I 731. My Calendar II 732. My Calendar III

My Calendar的book方法实现指定开始时间、结束时间&#xff0c;在重叠次数要求不同的情况下怎么实现。 729 My Calendar I 要求任意两个事件之间不能重叠。如果要插入的事件和已经插入的事件不重叠&#xff0c;则插入&#xff1b;否则不插入。 731 MyCalendar II 要求任意三个…

[转载]抓大放小,要事为先

最近看到了swing框架&#xff0c;细节颇多。细细看来效率很低&#xff0c;偶然找到了这篇文章&#xff0c;恩确实有道理&#xff1b;以下&#xff1b; 对待人生的任何事情都要&#xff1a;抓大放小&#xff0c;要事为先 对于一个以前从来没有接触过java的人&#xff0c;java无疑…

第三十七期:如果你这样回答“什么是线程安全”,面试官都会对你刮目相看

6月12日 以下文章来源于编程新说 &#xff0c;作者编程新说李新杰 有读者跟我说&#xff0c;喜欢看我的文章&#xff0c;说很容易读&#xff0c;我确实在易读性上花费的心思不亚于在内容上。因为我不喜欢一上来就堆很多东西&#xff0c;而且把简单的东西搞得复杂人人都会&…