给定一组字符串,你需要找到这组字符串中最长的非公共子序列。 最长的非公共子序列被定义为这些字符串之一的最长子序列,并且此子序列不应该是其他字符串的子序列。
子序列是可以通过删除一些字符而不改变其余元素的顺序从一个序列导出的序列。可以说,任何字符串都是自身的子序列,空字符串是任何字符串的子序列。
输入将是字符串列表,输出需要是最长的非公共子序列的长度。 如果最长的非公共子序列不存在,则返回-1。
这题很容易搞错,题目有点难以理解
要搞清楚子序列和子串的区别!子序列不需要连续,所以用str.contains(str2)是不行的
通过前面的题最长非公共子序列之1 我们知道,只要两个字符串不相等,那最长的字符串就是所求的解
但是我们不能用同样的思路来做这题,不能因为两个字符串相等就把它删掉,因为其他小的字符串可能是他们的子序列,也是不能作为解的
其实这题的做法有点类似于Brute force,直接判断每个子串是不是其他字符串的子序列,最长的子串满足这个要求的就是解。
没什么花样,只是理解题意上比较困难,也容易想走捷径
判断是不是子序列的基本功: 双指针
private boolean isSubsequence(String a, String b) {//a是b的子序列吗?
int i = 0;
for(int j = 0; i < a.length() && j < b.length(); j++) {
if(a.charAt(i) == b.charAt(j)) {
i++;
}
}
return i == a.length();
}
```
如果a比b的长度大,那a就不可能是b的子序列,这个判断我放在for 循环里了:
```
public int findLUSlength(String[] strs) {
Arrays.sort(strs, (a, b) -> b.length() - a.length());//从长到短排列
for(int i = 0; i < strs.length; i++) {
boolean flag = true;
for(int j=0; j < strs.length && strs[i].length() <= strs[j].length(); j++) {
if(i == j) continue;
if(isSubsequence(strs[i], strs[j])) {
flag = false;
break;
}
}
if(flag) {
return strs[i].length();
}
}
return -1;
}
```