刷题升级之路:Leetcode——522.最长特殊序列 II

题目描述

给定字符串列表 strs ,返回其中 最长的特殊序列。如果最长特殊序列不存在,返回 -1 。
特殊序列 定义如下:该序列为某字符串 独有的子序列(即不能是其他字符串的子序列)。
s的子序列可以通过删去字符串s中的某些字符实现。
例如,“abc” 是 “aebdc” 的子序列,因为您可以删除"aebdc"中的下划线字符来得到 “abc” 。“aebdc"的子序列还包括"aebdc”、 “aeb” 和 “” (空字符串)。

示例 1:
输入: strs = [“aba”,“cdc”,“eae”]
输出: 3

示例 2:
输入: strs = [“aaa”,“aaa”,“aa”]
输出: -1

提示:
2 <= strs.length <= 50
1 <= strs[i].length <= 10
strs[i] 只包含小写英文字母

来源:力扣(LeetCode)

【本题是Leecode521.的进阶版,需要更多的思考,在实现上也有不同方法。我和Leecode提供的题解就采用了不同方法】
我讲解Leecode521题的文章链接:link

算法思路和分析

思路一:Leecode题解算法

对于给定的某个字符串]str[i],如果它的一个子序列 sub 是「特殊序列」,那么str[i] 本身也是一个「特殊序列」。这是因为如果 sub 如果没有在除了str[i] 之外的字符串中以子序列的形式出现过,那么给 sub 不断地添加字符,它也不会出现。而str[i] 就是 sub 添加若干个(可以为零个)字符得到的结果。(这一点很容易理解,这种贪心的想法也是Leecode521题的基本认识)
因此我们只需要使用一个双重循环,外层枚举每一个字符串 str[i] 作为特殊序列,内层枚举每一个字符串,判断 str[i] 是否不为 str[j] 的子序列即可。(这是本算法的核心框架)
要想判断 str[i] 是否为 str[j] 的子序列,我们可以使用贪心 + 双指针的方法匹配。(匹配方法我的想法和题解基本一致,见下面的实现)。
来源:力扣(LeetCode)

上述算法的最坏时间复杂度:(n为字符串个数,m为最长字符串长度)
O(n²)阶字符串对数,每次匹配O(m)时间
O(mn²)

思路二:我的想法

我的想法是这样的,在Leecode521题中,即只有两个字符串的情形下,检查是否相等后,我们只需要取较长字符串即可。那么在n个字符串的情况下,优先检查更长的字符串是否可行,如果找到最长特殊序列,就不需要继续了,保证算法的高效性。
基于这样的想法,我的算法如下:
首先对strs中的字符串排序(先按照字符串从长到短,后按照字母顺序从大到小)
找到当前最长字符串,如果没有与之相等的,找到答案,直接返回其长度(一定最优);否则这些相等串都用不了,找剩余字符串中最长的串且不能是已经找过的字符串的子序列(否则也不能用),直到结束或者找到符合要求的字符串,返回其长度。
下面附上代码:

class Solution {
public:
    struct Sort{
        bool operator()(const string& a, const string& b){
            if(a.length()>b.length())
                return 1;
            else if(a.length()<b.length())
                return 0;
            return a>b;
        }
    };
    bool match(string a, string b){ // 查看b是否是a的子序列: 贪心法匹配
        int n1 = a.length(),n2 = b.length();
        int p1=0,p2=0;
        while(p1 < n1 && p2 < n2){
            if(a[p1]==b[p2]){
                p1++,p2++;
            }
            else
                p1++;
        }
        if(p1 == n1 && p2 < n2)
            return 0;
        return 1;
    }
    int findLUSlength(vector<string>& strs) {
        int n = strs.size();
        vector<string> been;
        sort(strs.begin(),strs.end(),Sort()); // 先按照字符串长度从长到短,后按照字母顺序大小从大到小排序
        int p=0,p_end=0; // p: 当前检查的位置; p_end: 与p位置字符串连续相等的结尾
        int num=0;
        while(p<n-1){
            p_end = p;
            while(p_end<n-1 && strs[p_end]==strs[p_end+1]){
                p_end++;
            }
            if(p_end==p){ //没有与strs[p]相等的字符串
                int flag=1;
                for(int i=0;i<num;i++){
                    if(match(been[i],strs[p])){ //当前字符串是之前某个字符串的子序列
                        flag=0;
                        break;
                    }
                }
                if(flag)
                    return strs[p].length(); // 返回最长特殊序列长度
            }
            been.push_back(strs[p]); //将所有相等字符串插入一个到been
            num++;
            p = p_end+1;
        }
        if(p == n-1){
            int flag=1;
            for(int i=0;i<num;i++){
                if(match(been[i],strs[p])){ //当前字符串是之前某个字符串的子序列
                    flag=0;
                    break;
                }
            }
            if(flag)
                return strs[p].length(); // 返回最长特殊序列长度
        }
        return -1; // 不存在最长特殊序列
    }
};

算法复杂度:
首先排序,
其余时间最坏情况下O(mn)

(代码写的正确但并不优美,应该有可以优化的地方让代码更高效简洁,而且算法复杂度分析不确定是否严谨,欢迎评论补充,谢谢!)
希望可以对你有所帮助,我们一起变得更强!!!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值