bzoj3790 manacher算法+贪心

紧跟jk大佬的步伐 这道题哇 因为机器一能生成回文串 所以我们只要用manacher跑一遍求出q【i】这样就把问题转化成了类似线段覆盖的题目 贪心就好了 至于,BIT优化dp我不会所以直接贪心了 注意答案是所需线段减一 因为是合并两条需要一次二机器 当然我ans(答案)直接初始化了0 懒得减一了 剩下的看代码吧

#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int M=100007;
char s[M];
int p[M],sum,len;
struct node{int l,r;}e[M];
bool cmp(node a,node b){
    if(a.l!=b.l) return a.l<b.l;
    return a.r>b.r;
}
void add(int x,int y){sum++; e[sum].l=x; e[sum].r=y;}
void manacher(){
    len=strlen(s);
    memset(p,0,sizeof(p));
    for(int i=len;i>=0;i--) s[i*2+2]=s[i],s[i*2+1]='#';  s[0]='&';
    int id=0; len=len*2; 
    for(int i=1;i<=len;i++){
        if(id+p[id]>i) p[i]=min(p[id*2-i],id+p[id]-i);
        else p[i]=1;
        while(s[i-p[i]]==s[i+p[i]]) p[i]++;
        if(i+p[i]>id+p[id]) id=i;
        add(i-p[i]+1,i+p[i]-1);
    }
}
void push_ans(){
    int r=e[1].r,ans=0,k=2; 
    while(r<len){
        int now=r;
        while(k<=sum&&e[k].l<=r) now=max(now,e[k].r),k++;    
        ans++; r=now; 
    }
    printf("%d\n",ans);
}
int main()
{
    while(~scanf("%s",s)){
        sum=0; manacher();
        sort(e+1,e+1+sum,cmp);
        //for(int i=1;i<=sum;i++) printf("[%d %d]\n",e[i].l,e[i].r);
        push_ans(); 
    }
    return 0;
}
View Code

 

posted @ 2017-06-08 11:22  友人Aqwq  阅读(264)  评论(1编辑  收藏  举报