BZOJ2803[Poi2012]Prefixuffix——hash

BZOJ2803[Poi2012]Prefixuffix——hash

题目描述

对于两个串S1、S2,如果能够将S1的一个后缀移动到开头后变成S2,就称S1和S2循环相同。例如串ababba和串abbaab是循环相同的。
给出一个长度为n的串S,求满足下面条件的最大的L:
1. L<=n/2
2. S的L前缀和S的L后缀是循环相同的。

输入

第一行一个正整数n (n<=1,000,000)。第二行n个小写英文字母,表示串S。

输出

一个整数,表示最大的L。

样例输入

15
ababbabababbaab

样例输出

6
 
假设两个串是循环同构的,那么这两个串可以表示成x+y和y+x的形式(其中x,y为两个字符串)
设f[i]表示前后都去掉i个字符后能匹配的最长前后缀长度,即y的最长长度。
手画一下能够发现对于位于左一半的i和i+1,f[i]<=f[i+1]+2。如果大于了就说明f[i+1]还能更大。
那么就可以从中间向开头转移f[i],最后求出i+f[i]的最大值即可。这道题卡自然溢出。

#include<set>
#include<map>
#include<queue>
#include<stack>
#include<cmath>
#include<vector>
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#define ll long long
using namespace std;
ll h[1000010];
ll g[1000010];
ll base1[1000010];
ll base2[1000010];
char ch[1000010];
int mod=1e9+7;
int n;
int f[1000010];
int ans=0;
bool check(int x,int y,int l)
{
    ll s1,s2,t1,t2;
    s1=((h[x+l-1]-h[x-1]*base1[l])%mod+mod)%mod;
    s2=((g[x+l-1]-g[x-1]*base2[l])%mod+mod)%mod;
    t1=((h[y+l-1]-h[y-1]*base1[l])%mod+mod)%mod;
    t2=((g[y+l-1]-g[y-1]*base2[l])%mod+mod)%mod;
    if(s1==t1&&s2==t2)
    {
        return true;
    }
    return false;
}
int main()
{
    scanf("%d",&n);
    scanf("%s",ch+1);
    base1[0]=1;
    base2[0]=1;
    for(int i=1;i<=n;i++)
    {
        base1[i]=base1[i-1]*233%mod;
        base2[i]=base2[i-1]*2333%mod;
        h[i]=(h[i-1]*233+(ch[i]-96))%mod;
        g[i]=(g[i-1]*2333+(ch[i]-96))%mod;
    }
    for(int i=n/2;i>=1;i--)
    {
        f[i]=min(f[i+1]+2,(n-2*i)/2);
        while(f[i]&&(!check(i+1,n-f[i]+1-i,f[i])))
        {
            f[i]--;
        }
    }
    for(int i=1;i<=n/2;i++)
    {
        if(!check(1,n-i+1,i))
        {
            continue;
        }
        ans=max(ans,f[i]+i);
    }
    printf("%d",ans);
}

转载于:https://www.cnblogs.com/Khada-Jhin/p/9637166.html

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/101431.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • ExecuteNonQuery()方法

    ExecuteNonQuery()方法ExecuteNonQuery()方法对Update,Insert,Delete语句有效,对select无效using(varconn=newSqlConnection(connectio

    2022年7月3日
    22
  • json学习初体验–第三者jar包实现bean、List、map创json格式

    json学习初体验–第三者jar包实现bean、List、map创json格式

    2022年1月13日
    56
  • struts2标签具体解释

    struts2标签具体解释

    2021年12月15日
    41
  • 迅雷种子为什么php文件后缀,迅雷BT文件后缀是什么?[通俗易懂]

    迅雷种子为什么php文件后缀,迅雷BT文件后缀是什么?[通俗易懂]你是否正在寻找关于文件后缀的内容?让我把最实时的东西奉献给你:迅雷BT文件后缀是什么?BT是一个后缀名为.torrent的小文件,它里面保存了服务器地址、要下载的文件的大孝分成的块数以及各种下载参数设置,这个文件一般在20k-100k大小,可以把*.php直接改成*.torrent试试!要么就是文件制作出错!在去这个页面下载一次,当弹出迅雷下载的时候点取消.让Windows下载.会出现保存对话框…

    2025年8月11日
    4
  • java中jbpm工作流_安卓框架

    java中jbpm工作流_安卓框架JBPM工作流框架应用导入jar包jbpm案例中获取配置文件,并配置本地数据库创建流程,并进行相关修改流程及流程内任务等的草操作importjava.io.File;importjava.io.FileInputStream;importjava.io.FileOutputStream;importjava.io.IOException;importj

    2025年10月10日
    4
  • Linux基础语法

    Linux基础语法LInuxlinux一切皆文件读写(权限)入门概述我们为什么要学习Linuxlinux诞生了这么多年,以前还喊着如何能取代windows系统,现在这个口号已经小多了,任何事物发展都有其局限性都有其天花板。就如同在国内再搞一个社交软件取代腾讯一样,想想而已基本不可能,因为用户已经习惯于使用微信交流,不是说技术上实现不了解而是老百姓已经习惯了,想让他们不用,即使他们自己不用亲戚朋友还是要用,没有办法的事情。用习惯了windows操作系统,再让大家切换到别的操作系统基本上是不可能的事情,改变一

    2022年5月18日
    35

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号