leetcode第14题

简介: 我们把原来的数组分成两部分,求出左半部分的最长公共前缀,求出右半部分的最长公共前缀,然后求出的两个结果再求最长公共前缀,就是最后的结果了。求左半部分的最长公共前缀,我们可以继续把它分成两部分,按照上边的思路接着求。然后一直分成两部分,递归下去。直到该部分只有 1 个字符串,那么最长公共子串就是它本身了,直接返回就可以了。

image.png

top14

解法一 垂直比较

我们把所有字符串垂直排列,然后一列一列的比较,直到某一个字符串到达结尾或者该列字符不完全相同。

下边看一下我的代码,看起来比较多

//这个函数判断 index 列的字符是否完全相同publicbooleanisSameAtIndex(String[] strs, intindex) {
inti=0;
while (i<strs.length-1) {
if (strs[i].charAt(index) ==strs[i+1].charAt(index)) {
i++;
        } else {
returnfalse;
        }
    }
returntrue;
}
publicStringlongestCommonPrefix(String[] strs) {
if (strs.length==0)
return"";
//得到最短的字符串的长度intminLength=Integer.MAX_VALUE;
for (inti=0; i<strs.length; i++) {
if (strs[i].length() <minLength) {
minLength=strs[i].length();
        }
    }
intj=0;
//遍历所有列for (; j<minLength; j++) {
//如果当前列字符不完全相同,则结束循环if (!isSameAtIndex(strs, j)) {
break;
        }
    }
returnstrs[0].substring(0, j);
}

下边看一下,官方的代码

publicStringlongestCommonPrefix(String[] strs) {
if (strs==null||strs.length==0) return"";
//遍历所有列for (inti=0; i<strs[0].length() ; i++){
charc=strs[0].charAt(i); // 保存 i 列第 0 行的字符便于后续比较//比较第 1,2,3... 行的字符和第 0 行是否相等for (intj=1; j<strs.length; j++) {
/*** i == strs[j].length() 表明当前行已经到达末尾* strs[j].charAt(i) != c  当前行和第 0 行字符不相等* 上边两种情况返回结果*/if (i==strs[j].length() ||strs[j].charAt(i) !=c)
returnstrs[0].substring(0, i);             
        }
    }
returnstrs[0];
}

时间复杂度:最坏的情况就是 n 个 长度为 m 的完全一样的字符串,假设 S 是所有字符的和,那么 S = m * n,时间复杂度就是 O(S)。当然正常情况下并不需要比较所有字符串,最多比较 n * minLen 个字符就可以了。

空间复杂度:O(1),常数个额外空间。

解法二 水平比较

我们将字符串水平排列,第 0 个和第 1 个字符串找最长子串,结果为 leet,再把结果和第 2 个字符串比较,结果为 leet,再把结果和第 3 个字符串比较,结果为 lee,即为最终结果。

publicStringlongestCommonPrefix3(String[] strs) {
if (strs.length==0)
return"";
Stringprefix=strs[0]; // 保存结果// 遍历每一个字符串for (inti=1; i<strs.length; i++) {
// 找到上次得到的结果 prefix 和当前字符串的最长子串intminLen=Math.min(prefix.length(), strs[i].length());
intj=0;
for (; j<minLen; j++) {
if (prefix.charAt(j) !=strs[i].charAt(j)) {
break;
                }
            }
prefix=prefix.substring(0, j);
        }
returnprefix;
    }

时间复杂度:最坏情况和解法一是一样,n 个长度为 m 的完全相同的字符,就要比较所有的字符 S,S = n * m 。但对于正常情况,处于最短字符串前的字符串依旧要比较所有字符,而不是最短字符串个字符,相对于解法一较差。

空间复杂度:O(1)。

解法三 递归

我们把原来的数组分成两部分,求出左半部分的最长公共前缀,求出右半部分的最长公共前缀,然后求出的两个结果再求最长公共前缀,就是最后的结果了。

求左半部分的最长公共前缀,我们可以继续把它分成两部分,按照上边的思路接着求。然后一直分成两部分,递归下去。

直到该部分只有 1 个字符串,那么最长公共子串就是它本身了,直接返回就可以了。

publicStringlongestCommonPrefix(String[] strs) {
if (strs==null||strs.length==0) return"";    
returnlongestCommonPrefix(strs, 0 , strs.length-1);
}
//递归不断分成两部分privateStringlongestCommonPrefix(String[] strs, intl, intr) {
if (l==r) {
returnstrs[l];
    }
else {
intmid= (l+r)/2;
StringlcpLeft=longestCommonPrefix(strs, l , mid);
StringlcpRight=longestCommonPrefix(strs, mid+1,r);
returncommonPrefix(lcpLeft, lcpRight);
   }
}
//求两个结果的最长公共前缀StringcommonPrefix(Stringleft,Stringright) {
intmin=Math.min(left.length(), right.length());       
for (inti=0; i<min; i++) {
if ( left.charAt(i) !=right.charAt(i) )
returnleft.substring(0, i);
    }
returnleft.substring(0, min);
}

时间复杂度:

空间复杂度:

每次遇到递归的情况,总是有些理不清楚,先空着吧。

进行了垂直比较和水平比较,又用到了递归,solution 里还介绍了二分查找,感觉这里用二分查找有些太僵硬了,反而使得时间复杂度变高了。还介绍了前缀树,这里后边遇到再总结吧。


相关文章
|
4月前
|
算法
LeetCode第66题加一
LeetCode第66题"加一"的解题方法,通过遍历数组从后向前处理每一位的加法,并考虑进位情况,最终实现给定数字加一的功能。
LeetCode第66题加一
|
7月前
leetcode-827:最大人工岛
leetcode-827:最大人工岛
64 0
|
7月前
leetcode-1219:黄金矿工
leetcode-1219:黄金矿工
84 0
|
7月前
leetcode-475:供暖器
leetcode-475:供暖器
50 0
|
7月前
|
消息中间件 Kubernetes NoSQL
LeetCode 1359、1360
LeetCode 1359、1360
|
存储 Python
LeetCode 66. Plus One
给定表示非负整数的非空数字数组,加上整数的1。 存储数字使得最高有效数字位于列表的开头,并且数组中的每个元素包含单个数字。 您可以假设整数不包含任何前导零,除了数字0本身。
92 0
LeetCode 66. Plus One
|
存储
leetcode第57题
和上一道可以说是一个问题,只不过这个是给一个已经合并好的列表,然后给一个新的节点依据规则加入到合并好的列表。 解法一 对应 56 题的解法一,没看的话,可以先过去看一下。这个问题其实就是我们解法中的一个子问题没看的话,可以先过去看一下。这个问题其实就是我们解法中的一个子问题, 所以直接加过来就行了
105 0
leetcode第57题
|
机器学习/深度学习
leetcode第50题
求幂次方,用最简单的想法,就是写一个 for 循环累乘。 至于求负幂次方,比如 2^{-10}2−10,可以先求出 2^{10}210,然后取倒数,1/2^{10}1/210 ,就可以了 double mul = 1; if (n > 0) { for (int i = 0; i < n; i++) { mul *= x; } } else { n = -n; for (int i = 0; i < n; i++) { mul *= x; } mul = 1 / mul; }
leetcode第50题
leetcode第44题
时间复杂度:text 长度是 T,pattern 长度是 P,那么就是 O(TP)。 空间复杂度:O(TP)。 同样的,和第10题一样,可以优化空间复杂度。
leetcode第44题
|
算法
leetcode第30题
如上图,利用循环变量 i ,依次后移,判断每个子串是否符合即可。 怎么判断子串是否符合?这也是这个题的难点了,由于子串包含的单词顺序并不需要固定,如果是两个单词 A,B,我们只需要判断子串是否是 AB 或者 BA 即可。如果是三个单词 A,B,C 也还好,只需要判断子串是否是 ABC,或者 ACB,BAC,BCA,CAB,CBA 就可以了,但如果更多单词呢?那就崩溃了。 链接)的作者提出了,用两个 HashMap 来解决。首先,我们把所有的单词存到 HashMap 里,key 直接存单词,value 存单词出现的个数(因为给出的单词可能会有重复的,所以可能是 1 或 2 或者其他)。然后扫描子
124 0
leetcode第30题