Codeforces 319D Have You Ever Heard About the Word?

首先会想到|x|是不递减的。

于是可以枚举长度L。

再每个L设一个断点,xx必定经过两个断点。

两两断点间求最长公共前后缀,这里用hash+二分会快。

然后一波扫过去就好了。

如果找到了,hash就要重构。

来计算一下复杂度。

一共有O(n log n)个断点,每个求最长公共前后缀复杂度log,这一部分是O(n log^2 n )
长度小于 \(L \leq n\)的最多 \(n\sqrt n\)次,\(L\geq n\)最多 \(n\sqrt n\)中,所以重构复杂度: \(O(n \sqrt n)\)

#include<cstdio>
#include<cstring>
#include<algorithm>
#define ll long long
#define fo(i, x, y) for(int i = x; i <= y; i ++)
#define min(a, b) ((a) < (b) ? (a) : (b))
#define P pair<int, int>
using namespace std;

const int N = 50005, mo = 998244353, pri = 1e9 + 7, pri2 = 43313;

char str[N];
int n, bz[N], tmp;
ll c[N], ni[N], s[N];

ll ksm(ll x, ll y) {
    ll s = 1;
    for(; y; x = x * x % mo, y >>= 1)
        if(y & 1) s = s * x % mo;
    return s;
}

void Getsum() {
    fo(i, 1, n) s[i] = (s[i - 1] + c[i] * (str[i] - 'a') % mo) % mo;
}

int sum(int x, int y) {
    return ((s[y] - s[x - 1] + mo) * ni[x] % mo);
}

int Getq(int x, int y) {
    int ans = 0;
    for(int l = 1, r = tmp; l <= r;) {
        int m = l + r >> 1;
        if(sum(x - m + 1, x) == sum(y - m + 1, y))
            ans = m, l = m + 1; else r = m - 1;
    }
    return ans;
}

int Geth(int x, int y) {
    int ans = 0;
    for(int l = 1, r = n - y + 1; l <= r;) {
        int m = l + r >> 1;
        if(sum(x, x + m - 1) == sum(y, y + m - 1))
            ans = m, l = m + 1; else r = m - 1;
    }
    return ans;
}

int main() {
    scanf("%s", str + 1); n = strlen(str + 1);

    c[0] = ni[0] = 1;
    ni[1] = ksm(pri, mo - 2); c[1] = pri;
    fo(i, 2, n) ni[i] = ni[i - 1] * ni[1] % mo, c[i] = c[i - 1] * c[1] % mo;

    int n0 = n; Getsum();
    fo(l, 1, n) {
        int xg = 0; tmp = l;
        fo(i, 1, n / l) {
            int x = i * l, y = x + l;
            if(y > n) break;
            int q = Getq(x, y), h = Geth(x, y);
            if(q + h > l) {
                fo(j, x - q + 1, x - q + l) bz[j] = l;
                tmp = q;
                xg = 1;
            } else tmp = l;
        }
        if(xg) {
            int n1 = 0;
            fo(i, 1, n) if(bz[i] != l)
                str[++ n1] = str[i];
            n = n1;
            Getsum();
        }
    }
    fo(i, 1, n) putchar(str[i]);
}

转载于:https://www.cnblogs.com/wawawa8/p/10840061.html

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
提供的源码资源涵盖了安卓应用、小程序、Python应用和Java应用等多个领域,每个领域都包含了丰富的实例和项目。这些源码都是基于各自平台的最新技术和标准编写,确保了在对应环境下能够无缝运行。同时,源码中配备了详细的注释和文档,帮助用户快速理解代码结构和实现逻辑。 适用人群: 这些源码资源特别适合大学生群体。无论你是计算机相关专业的学生,还是对其他领域编程感兴趣的学生,这些资源都能为你提供宝贵的学习和实践机会。通过学习和运行这些源码,你可以掌握各平台开发的基础知识,提升编程能力和项目实战经验。 使用场景及目标: 在学习阶段,你可以利用这些源码资源进行课程实践、课外项目或毕业设计。通过分析和运行源码,你将深入了解各平台开发的技术细节和最佳实践,逐步培养起自己的项目开发和问题解决能力。此外,在求职或创业过程中,具备跨平台开发能力的大学生将更具竞争力。 其他说明: 为了确保源码资源的可运行性和易用性,特别注意了以下几点:首先,每份源码都提供了详细的运行环境和依赖说明,确保用户能够轻松搭建起开发环境;其次,源码中的注释和文档都非常完善,方便用户快速上手和理解代码;最后,我会定期更新这些源码资源,以适应各平台技术的最新发展和市场需求。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值