最长子序列最长子串的题型汇总

article/2025/9/28 10:05:45

1.最长公共子序列的长度

题目:对于两个字符串,请设计一个高效算法,求他们的最长公共子序列的长度,这里的最长公共子序列定义为有两个序列U1,U2,U3...Un和V1,V2,V3...Vn,其中Ui&ltUi+1,Vi&ltVi+1。且A[Ui] == B[Vi]。

给定两个字符串AB,同时给定两个串的长度nm,请返回最长公共子序列的长度。保证两串长度均小于等于300。

样例:

"1A2C3D4B56",10,"B1D23CA45B6A",12
返回:

6

解析:求不连续的公共子序列的长度,O(n2)

dp[ i ][ j ] 为A的0 - i 的子串与B的0 - i 的子串的最大公共子序列长度

// 最长公共子序列,动规,O(n2)public static int findLCS(String A, String B) {int n = A.length();int m = B.length();int[][] dp = new int[n + 1][m + 1];for (int i = 0; i < n; ++i) {for (int j = 0; j < m; ++j) {if (A.charAt(i) == B.charAt(j)) {dp[i + 1][j + 1] = dp[i][j] + 1;} else {dp[i + 1][j + 1] = Math.max(dp[i][j + 1], dp[i + 1][j]);}}}return dp[n][m];}

2.打印最长公共子序列

同上,回溯打印。

	// 打印最长公共子序列,动规,O(n2)public static String findLCS2(String A, String B) {int n = A.length();int m = B.length();int[] path = new int[n];StringBuffer ret = new StringBuffer();int[][] dp = new int[n + 1][m + 1];for (int i = 0; i < n; ++i) {for (int j = 0; j < m; ++j) {if (A.charAt(i) == B.charAt(j)) {dp[i + 1][j + 1] = dp[i][j] + 1;} else {dp[i + 1][j + 1] = Math.max(dp[i][j + 1], dp[i + 1][j]);}}}int i = n;int j = m;int k = 0;while (dp[i][j] > 0) {if (dp[i][j] == dp[i - 1][j])i--;else if (dp[i][j] == dp[i][j - 1])j--;else {path[k++] = i - 1;i--;j--;}}for (i = k - 1; i >= 0; i--)ret.append(A.charAt(path[i]));return ret.toString();}

3.最长公共子串的长度

题目:对于两个字符串,请设计一个时间复杂度为O(m*n)的算法(这里的m和n为两串的长度),求出两串的最长公共子串的长度。这里的最长公共子串的定义为两个序列U1,U2,..Un和V1,V2,...Vn,其中Ui + 1 == Ui+1,Vi + 1 == Vi+1,同时Ui == Vi。

给定两个字符串AB,同时给定两串的长度nm

样例:

"1AB2345CD",9,"12345EF",7

返回:4

解析:求连续的公共子串的长度,O(n2)

dp[ i ][ j ] 为A的以 i 结尾的和子串与B的以 j 结尾的子串最大公共子串长度

// 最长公共子串,动规,O(n2)public int findLongest(String A, String B) {int n = A.length();int m = B.length();int[][] dp = new int[n][m];int max = 0;for (int i = 0; i < n; i++) {for (int j = 0; j < m; j++) {if (A.charAt(i) == B.charAt(j)) {if (i == 0 || j == 0) {dp[i][j] = 1; // if(arrA[i]==arrB[0])dp[i][0]=1;if(arrB[j]==arrA[0])dp[0][j]=1;} else {dp[i][j] = dp[i - 1][j - 1] + 1;}max = Math.max(dp[i][j], max);}}}return max;}

4.打印最长公共子串 

例子同上,稍作修改便可:记录最大连续结尾位置

// 打印最长公共子串,动规,O(n2)public static String findLongest2(String A, String B) {int n = A.length();int m = B.length();int index_i = 0;// 记录最长公共子串结尾位置int[][] dp = new int[n][m];int max = 0;for (int i = 0; i < n; i++) {for (int j = 0; j < m; j++) {if (A.charAt(i) == B.charAt(j)) {if (i == 0 || j == 0) {dp[i][j] = 1; // if(arrA[i]==arrB[0])dp[i][0]=1;if(arrB[j]==arrA[0])dp[0][j]=1;} else {dp[i][j] = dp[i - 1][j - 1] + 1;}if (max < dp[i][j]) {max = dp[i][j];index_i = i;}}}}String result = A.substring(index_i - max + 1, index_i + 1);return result;}

5.A串变为B串所需要的最少代价

题目:对于两个字符串A和B,我们需要进行插入、删除和修改操作将A串变为B串,定义c0,c1,c2分别为三种操作的代价,请设计一个高效算法,求出将A串变为B串所需要的最少代价。
给定两个字符串A和B,及它们的长度和三种操作代价,请返回将A串变为B串所需要的最小代价。保证两串长度均小于等于300,且三种代价值均小于等于100。
样例:
        "abc",3,"adc",3,5,3,100
返回:8

解析:

dp[i][j]表示A[0..i-1]变到B[0..j-1]需要的最小代价

长度为i的A修改为长度为j的B可以分为:
1、长度为i的A修改为长度为j-1的B,然后插入j位置的字符;
2、长度为i-1的A修改为长度为j的B,然后删除i位置的字符;
3、长度为i-1的A修改为长度为j-1的B,然后i位置的字符修改为j位置的字符。

public static int Str1ToStr2(String A, String B, int c0, int c1, int c2) {int n = A.length();int m = B.length();int[][] dp = new int[n + 1][m + 1];// dp[i][j]表示A[0..i-1]变到B[0..j-1]需要的最小代价for (int i = 1; i < n + 1; i++) {dp[i][0] = dp[i - 1][0] + c1;// 删除}for (int j = 1; j < m + 1; j++) {dp[0][j] = dp[0][j - 1] + c0;// 插入}for (int i = 1; i < n + 1; i++) {for (int j = 1; j < m + 1; j++) {if (A.charAt(i - 1) == B.charAt(j - 1)) {dp[i][j] = dp[i - 1][j - 1];} else {int cost1 = dp[i][j - 1] + c0;// 插入时的代价int cost2 = dp[i - 1][j] + c1;// 删除的代价int cost3 = dp[i - 1][j - 1] + c2;// 修改的代价dp[i][j] = Math.min(cost3, Math.min(cost1, cost2));}}}return dp[n][m];}

6.由两个字符串交错组成

题目:对于三个字符串A,B,C。我们称C由A和B交错组成当且仅当C包含且仅包含A,B中所有字符,且对应的顺序不改变。请编写一个高效算法,判断C串是否由A和B交错组成。

给定三个字符串A,BC,及他们的长度。请返回一个bool值,代表C是否由A和B交错组成。保证三个串的长度均小于等于100。

样例:

    "ABC",3,"12C",3,"A12BCC",6
返回:true

解析:判断C串是否由A和B交错组成。

dp[ i ][ j ] : A的前个字符与B的前j个字符是否与C的第前i+j-1个字符匹配

public static boolean chkMixture(String A, String B, String C) {char[] a = A.toCharArray();char[] b = B.toCharArray();char[] c = C.toCharArray();int n = a.length;int m = b.length;int v = c.length;if (m + n != v) {return false;}boolean[][] dp = new boolean[n + 1][m + 1];// A的前个字符与B的前j个字符是否与C的第前i+j-1个字符匹配dp[0][0] = true;for (int i = 1; i <= n; i++) {if (a[i - 1] == c[i - 1]) {// 只有A的字符串与C匹配dp[i][0] = true;} else {break;}}for (int j = 1; j <= m; j++) {if (b[j - 1] == c[j - 1]) {// 只有B的字符串与C匹配dp[0][j] = true;} else {break;}}for (int i = 1; i < n + 1; i++) {for (int j = 1; j < m + 1; j++) {if (dp[i - 1][j] && a[i - 1] == c[i + j - 1]) {dp[i][j] = true;continue;}if (dp[i][j - 1] && b[j - 1] == c[i + j - 1]) {dp[i][j] = true;}}}return dp[n][m];}

7.无重复字符的最长子串

题目:给定一个字符串,找出不含有重复字符的 最长子串 的长度。

样例:

        给定 "abcabcbb" ,没有重复字符的最长子串是 "abc" ,那么长度就是3。

解析:

1. 建立一个256位大小的整型数组来代替哈希表,这样做的原因是ASCII表共能表示256个字符,记录所有字符;

2. 定义两个变量res和left,其中res用来记录最长无重复子串的长度,left指向该无重复子串左边的起始位置,遍历整个字符串,对于每一个遍历到的字符:

  • 如果哈希表中该字符串对应的值为0,说明没有遇到过该字符,则此时计算最长无重复子串,i - left +1,其中i是最长无重复子串最右边的位置,left是最左边的位置;
  • 还有一种情况也需要计算最长无重复子串,就是当哈希表中的值小于left,这是由于此时出现过重复的字符,left的位置更新了,如果又遇到了新的字符,就要重新计算最长无重复子串。

3. 最后每次都要在哈希表中将当前字符对应的值赋值为i+1。

复杂度O(n)

public static int lengthOfLongestSubstring(String s) {char[] a = s.toCharArray();int[] m = new int[256];int res = 0, left = 0;for (int i = 0; i < s.length(); ++i) {if (m[a[i]] == 0 || m[a[i]] < left) {res = Math.max(res, i - left + 1);} else {left = m[a[i]];}m[a[i]] = i + 1;}return res;}

改进算法:

public static int lengthOfLongestSubstring2(String s) {char[] a = s.toCharArray();int[] m = new int[256];int res = 0, left = 0;for (int i = 0; i < s.length(); ++i) {left = Math.max(left, m[a[i]]);m[a[i]] = i;res = Math.max(res, i - left);}return res;}

8.求和为0的最长连续子数组

解析:

  通过分析可知,要使其和为0,只有当1和-1的个数相等时,才会成立,但题目要求是连续子序列,所以单纯统计其1和-1个数不可取。

  由题目中求最长连续子序列,可想到动态规划来求解,动态规划的求解既是寻找其状态转移方程和建立状态转移表的过程

  设dp[i]为下标为i及其之前数组中所有元素的和,

                                               

如图所示,数组为1,-1,1,-1,1,-1,1,-1最后一个值为0,直接满足结果,输出8

如上图,数组1,1,-1,1,1,-1,-1,dp取值为dp[0] = dp[2] = dp[6] = 1; dp[1] = dp[3] = d[5] = 3; dp[4] = 3;

对于每个值,取最后一次出现的位置和第一次出现的位置之差,取它们的最大值,max((6 - 0),(5 - 1),(4 - 4) = 6

private static int solve(int[] numbers, int n) {int[] dp = new int[n];dp[0] = numbers[0];for (int i = 1; i < n; i++) {dp[i] = dp[i - 1] + numbers[i];}Map<Integer, Integer> dpMap = new HashMap<>();dpMap.put(0, 0);int maxLen = 0;for (int i = 0; i < n; i++) {if (!dpMap.containsKey(dp[i])) {dpMap.put(dp[i], i + 1);} else {int len = i - dpMap.get(dp[i]) + 1;maxLen = maxLen < len ? len : maxLen;}}return maxLen;}

9.和不大于M的最大连续数列。 

双指针法  

// 和不大于M的最大连续子数列。public static int[] findSubMaxSum(int[] numbers, int M) {int[] cumSum = new int[numbers.length + 1]; // obtain cumulative sum.int sum = 0;cumSum[0] = 0;for (int i = 0; i < numbers.length; i++) {sum += numbers[i];cumSum[i + 1] = sum;}int l = 0, r = 0; // two pointers start at tip of the array.int max = 0;int[] ids = new int[2];while (l < cumSum.length) {while (r < cumSum.length && cumSum[r] - cumSum[l] <= M) {r++;}if (cumSum[r - 1] - cumSum[l] > max) { // since cumSum[0] = 0, thus r always > 0.max = cumSum[r - 1] - cumSum[l];ids[0] = l;ids[1] = r;}l++;}System.out.println(max);return ids;}

10.至少含有K个不同数字的最小连续数列 

双指针法 

// 至少含有K个不同数字的最小连续子数列public static int[] findSubMinSum(int[] numbers, int K) {int[] cumSum = new int[numbers.length + 1]; // obtain cumulative sum.int sum = 0;cumSum[0] = 0;for (int i = 0; i < numbers.length; i++) {sum += numbers[i];cumSum[i] = sum;}int l = 0, r = 0; // two pointers start at tip of the array.int min = Integer.MAX_VALUE;int[] ids = new int[2];HashSet<Integer> set = new HashSet<>();while (l < numbers.length) {while (r < numbers.length && set.size() < K) {set.add(numbers[r]);r++;}int sumStart = l > 0 ? cumSum[l - 1] : 0;if (cumSum[r - 1] - sumStart < min) {min = cumSum[r - 1] - sumStart;ids[0] = l + 1;ids[1] = r;}set.remove(numbers[l]);while (l + 1 < numbers.length && numbers[l] == numbers[l + 1]) {l++;}l++;}System.out.println(min);return ids;}

 

 

 

 


http://chatgpt.dhexx.cn/article/ABc9Q3EG.shtml

相关文章

动态规划解最长子序列问题

动态规划法 经常会遇到复杂问题不能简单地分解成几个子问题&#xff0c;而会分解出一系列的子问题。简单地采用把大问题分解成子问题&#xff0c;并综合子问题的解导出大问题的解的方法&#xff0c;问题求解耗时会按问题规模呈幂级数增加。 为了节约重复求相同子问题的时间&…

最长****子序列

&#xff08;在研读大佬写的博客后&#xff0c;打算记录下自己学习过程&#xff09; 通过最长上升子序列的拓展了解到&#xff0c;其实这是一个系列的问题主要代表有&#xff1a; 1 最长上升子序列 2 最长不上升子序列 3 最长下降子序列 4 最长不下降子序列 就以最长上升子…

最长公共子序列

最长公共子序列&#xff08;Longest Common Subsequence&#xff0c;简称 LCS&#xff09;是一道非常经典的面试题目&#xff0c;因为它的解法是典型的二维动态规划&#xff0c;大部分比较困难的字符串问题都和这个问题一个套路&#xff0c;比如说编辑距离。而且&#xff0c;这…

最长公共子序列(LCS) 过程图解

1.基本概念 首先需要科普一下&#xff0c;最长公共子序列&#xff08;longest common sequence&#xff09;和最长公共子串&#xff08;longest common substring&#xff09;不是一回事儿。什么是子序列呢&#xff1f;即一个给定的序列的子序列&#xff0c;就是将给定序列中零…

最长子序列问题详解

提到最长子序列问题&#xff0c;想必大家都不陌生&#xff0c;今天我主要想分享一下我对子序列问题的一些理解&#xff1a; 先拿最长上升子序列问题来说吧&#xff1a; 很明显这是一个动态规化问题&#xff0c;仔细想想也不难得出其状态转移方程 首先介绍一下dp[]数组的含义…

如何查看pip版本

Windows系统如何查看pip版本 直接运行pip show pip

python之pip版本问题

我们在默认安装python软件之后&#xff0c;其自行安装的pip的版本可能比较落后。我们使用pip安装模块的时候经常提示安装失败&#xff0c;查看安装失败原因&#xff0c;部分是因为pip不是最新版本需要升级版本才能继续安装&#xff0c;因此写下一点小心得&#xff0c;让我们都能…

pip 查看可安装版本

pip 查看可安装版本 pip版本&#xff1a; pip -V pip 20.1.1 from /usr/lib/python2.7/site-packages/pip (python 2.7) 查看pip可安装版本 import pip._internal.utils.compatibility_tags print(pip._internal.utils.compatibility_tags.get_supported())

python如何查看pip版本并且升级pip

第一次写Python的学习经历 我之前也安装过Python,今天&#xff0c;终于重新安装了64位的windows的Python,于是在命令行输入&#xff1a;“pip list”出现以下的提示&#xff1a; 这个提示以前也出现过&#xff0c;但是看不懂&#xff0c;也不知道怎么处理&#xff0c;然后又胡…

python pip 查找指定安装包版本信息

一、windows操作系统 pip list | findstr 查找的包&#xff08;支持模糊查询&#xff09;例&#xff1a; pip list | findstr huawei结果&#xff1a; 二、linux操作系统 pip list | grep 查找的包&#xff08;支持模糊查询&#xff09;

更新pip版本

更新pip版本 从报错来看是因为pip版本为20.2.3&#xff0c;但当前可用版本是21.0.1 python -m ensurepippython -m pip install --upgrade pippip list设置镜像默认源 #临时使用 pip install -i https://pypi.tuna.tsinghua.edu.cn/simple some-package #设置默认源 pip inst…

拉取的docker镜像中更换python以及pip版本

场景&#xff1a; 需要tensorflow1.10 python3.6 的环境&#xff0c;但是拉取的tf1.10镜像中&#xff0c;默认python版本是3.5 &#xff0c;此时我们需要安装python3.6版本&#xff0c;将python命令的软连接删除&#xff0c;并且指向新安装的3.6&#xff0c;且pip的连接也应该…

Python查看所有版本及pip

1、查看python默认版本&#xff1a; python --version &#xff08;默认的版本我改为了python3&#xff0c;所以结果是3.7的&#xff09; 2、查看另外安装的2.7的版本&#xff1a; python2.7 --version 3、查看python安装路径&#xff1a; 1&#xff09;终端输入python2.7回…

python降低pip版本

场景&#xff1a; 使用Pycharm开发时&#xff0c;经常会遇到因为pip版本过高而导致下载第三方库失败的情况 解决方案&#xff1a; 经查资料&#xff0c;发现可以通过降低pip的版本来解决问题 如下&#xff1a; 第一种方法&#xff1a;使用Terminal命令行进行降级 python -m pip…

Linux查看pip版本号的命令,pip 经常使用命令及控制台怎么查看python 及pip 和已安装包版本号...

在使用python的时候&#xff0c;常常使用到pip这个工具&#xff0c;能够很方便的线上安装依赖库&#xff0c;固然pip还有不少参数均可以帮咱们去查询一些库信息&#xff0c;在安装python的时候&#xff0c;下载带有pip的安装包就能够直接安装pip啦&#xff0c;固然没有带pip的&…

查看pip版本 及升级pip命令报错解决办法

pip install pytest_ordering时警告&#xff0c;pip版本过低 WARNING: You are using pip version 21.1.3; however, version 21.2.4 is available. You should consider upgrading 1、查看pip&#xff1a; pip show pip 2、升级pip&#xff1a;python -m pip install --upg…

最新升级pip命令,查看pip版本命令,pycharm升级pip命令,推荐收藏关注不迷路

1. 查看pip版本命令&#xff1a;pip -V &#xff0c;可以看看以下代码运行完过后&#xff0c;自己的pip版本有没有升级成功 python升级pip命令&#xff1a;打开命令行输入 python -m pip install -U pip 回车 pycharm升级pip命令 &#xff1a;打开命令行输入&#xff1a; py…

[Python]pip查找包的历史版本

pip查找包的历史版本 场景&#xff1a;在一些时候通过pip install xxx 安装第三方库的时候默认情况下安装最新版本&#xff0c;由于是最新版本有个稳定性就不得不考虑其中&#xff0c;所以部分场景会存在一些bug这就要求我们安装历史版本&#xff0c;对一些更新频率比较高的三…

已解决正确查看pip版本、查看pip帮助命令

已解决&#xff08;pip查看版本报错&#xff09;Usage&#xff1a;pip [options] no such option: —verson 文章目录 报错代码报错翻译报错原因解决方法千人全栈VIP答疑群联系博主帮忙解决报错 报错代码 粉丝群一个小伙伴想用pip查看版本报错&#xff0c;但是发生了报错&#…