leetcode 1044. 最长重复子串—(每日一难day16)

1044. 最长重复子串

给你一个字符串 s ,考虑其所有 重复子串 :即 s 的(连续)子串,在 s 中出现 2 次或更多次。这些出现之间可能存在重叠。

返回 任意一个 可能具有最长长度的重复子串。如果 s 不含重复子串,那么答案为 “” 。

示例 1:

输入:s = “banana”
输出:“ana”

示例 2:

输入:s = “abcd”
输出:“”

提示:

2 <= s.length <= 3 * 1e4
s 由小写英文字母组成

来源:力扣(LeetCode)
链接:lianjie

解析

  • 首先,用二分的方法,枚举字符串的长度
  • 然后用双指针+hash方法,寻找相同的字符串
  • 需要用到字符串hash的技巧 相同技巧讲解

code

const int N = 3e4 + 10, P = 131;

typedef unsigned long long ULL;

class Solution {
private:
    string res = "";
    string s;
    int n;
    ULL h[N] = {0}, p[N] = {0};//h存储字符串哈希的前缀数组,p[i]表示P的i次方的值
public:
    string longestDupSubstring(string s) 
    {
        this -> s = s;
        n = s.size();
        p[0] = 1;
        for (int i = 1; i <= n; i ++ )//字符串哈希的前缀和
        {
            h[i] = h[i - 1] * P + s[i - 1];
            p[i] = p[i - 1] * P;
        }

        int l = 0, r = n;
        while (l < r)//二分查找最长重复串
        {
            int mid = l + r + 1 >> 1;
            if (check(mid)) l = mid;
            else r = mid - 1;
        }

        return res;
    }

    ULL get(int l, int r)//求出区间[l, r]的哈希值
    {
        return h[r] - h[l - 1] * p[r - l + 1];
    }

    bool check(int len)
    {
        unordered_set<ULL> S;//记录哈希值
        for (int i = len; i <= n; i ++ )
        {
            ULL num = get(i - len + 1, i);//求出哈希值
            if (S.count(num))//存在重复串
            {
                res = s.substr(i - len, len);//更新重复串并返回true
                return true;
            }
            else S.insert(num);//加入哈希表
        }

        return false;//不存在重复串
    }
};
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值