题目描述
14.最长公共前缀
编写一个函数来查找字符串数组中的最长公共前缀。
如果不存在公共前缀,返回空字符串 “”。
示例1
输入: [“flower”,“flow”,“flight”]
输出: “fl”
示例2
输入: [“dog”,“racecar”,“car”]
输出: “”
解释: 输入不存在公共前缀。
说明
所有输入只包含小写字母 a-z 。
解题思路
1.直接法(水平扫描)
- 先求出前两个字符串公共前缀 ans
- 再求 ans 和之后字符串的公共前缀
- 若公共字符串为 "" ,则直接 return ""
- 否则直至循环终止
- 时间复杂度: O(n),n为全部字符串长度和
- 所需空间: O(1)
代码如下
class Solution {
public:
string longestCommonPrefix(vector<string>& strs) {
if(strs.size() == 1) return strs[0];
if(strs.size() == 0) return "";
string ans = CommonPrefix(strs[0], strs[1]);
for(int i = 2; i < strs.size(); i++)
{
ans = CommonPrefix(ans, strs[i]);
if(ans == "") break;
}
return ans;
}
// 两字符串最长公共前缀
string CommonPrefix(string s1, string s2)
{
string ans = "";
int len = s1.length() < s2.length() ? s1.length() : s2.length();
for(int i = 0; i < len; i++)
{
if(s1[i] != s2[i])
{
break;
}
ans += s1[i];
}
return ans;
}
};
执行结果
2.分治算法
- 要求最长公共子序列
- 可将问题分为求 n 个相邻子序列的公共子序列
- 依次递归求出答案
- 时间复杂度:O(n)
- 所需空间:O(mlogn) 所需空间主要是递归开辟的栈空间
代码如下
class Solution {
public:
string longestCommonPrefix(vector<string>& strs) {
if(strs.size() == 1) return strs[0];
if(strs.size() == 0) return "";
return longestCommonPrefix(strs, 0 , strs.size() - 1);
}
string longestCommonPrefix(vector<string>& strs, int left, int right)
{
if(left == right)
{
return strs[left];
}
else
{
int mid = (left + right) / 2;
string lcommon = longestCommonPrefix(strs, left, mid);
string rcommon = longestCommonPrefix(strs, mid + 1, right);
return CommonPrefix(lcommon, rcommon);
}
}
// 两字符串最长公共前缀
string CommonPrefix(string s1, string s2)
{
string ans = "";
int len = s1.length() < s2.length() ? s1.length() : s2.length();
for(int i = 0; i < len; i++)
{
if(s1[i] != s2[i])
{
break;
}
ans += s1[i];
}
return ans;
}
};