P4762 [CERC2014]Virus synthesis

题意

真是道回文自动机好题。

首先考虑答案必定是一个回文串+剩余部分的形式,因此可以建出回文自动机,之后考虑每个长度为偶数的回文串。

对于一个长度为偶数的回文串,设它在回文自动机上对应的节点为\(x\),我们对于每个\(x\)求出\(trans_x\)表示x的最长后缀回文串,满足\(len_{trans_x}\leqslant len_x/2\)。

之后设\(f_x\)表示\(x\)拼成\(x\)这个串的最小代价,我们从\(0\)(偶根)出发进行\(bfs\),中途计算\(f_x\)。

对于\(f_x\):
初值肯定是自身长度\(f_x=len_x\)。
如果存在一条边\((x,y)\),那么\(f_y=f_x+1\),因为我们可以在拼\(x\)时还没进行\(2\)操作时向\(x\)后面填一个字符,使其进行\(2\)操作后变为\(y\)。
同时\(f_x=min(f_x,f_{trans_x}+1+lem_x/2-len_{trans_x})\),即我们可以从\(trans_x\)变过来。

对于每个\(x\),它对答案的贡献是\(n-len_x+f_x\),\(n\)是字符串长度。

code:

#include<bits/stdc++.h>
using namespace std;
const int maxn=1e5+10;
const int inf=1e9;
int T,n,tot,last,ans;
int fail[maxn],len[maxn],trans[maxn],f[maxn];
int ch[maxn][5];
char s[maxn];
inline int change(char c)
{
    if(c=='A')return 1;
    if(c=='G')return 2;
    if(c=='C')return 3;
    if(c=='T')return 4;
    return 2333;
}
inline void init()
{
    for(int i=0;i<=tot;i++)
        for(int j=1;j<=4;j++)
            ch[i][j]=0;
    fail[0]=1;len[1]=-1;
    tot=1;last=0;
}
inline int getfail(int x,int pos)
{
    while(s[pos-len[x]-1]!=s[pos])x=fail[x];
    return x;
}
inline void add(int c,int pos)
{
    int p=getfail(last,pos);
    if(!ch[p][c])
    {
        int q=++tot,tmp;len[q]=len[p]+2;
        tmp=getfail(fail[p],pos);
        fail[q]=ch[tmp][c];ch[p][c]=q;
        if(len[q]<=2)trans[q]=fail[q];
        else
        {
            tmp=trans[p];
            while(s[pos-len[tmp]-1]!=s[pos]||((len[tmp]+2)<<1)>len[q])tmp=fail[tmp];
            trans[q]=ch[tmp][c];
        }
    }
    last=ch[p][c];
}
inline void solve()
{
    queue<int>q;
    for(int i=2;i<=tot;i++)f[i]=len[i];
    for(int i=1;i<=4;i++)if(ch[0][i])q.push(ch[0][i]);
    while(!q.empty())
    {
        int x=q.front();q.pop();
        f[x]=min(f[x],f[trans[x]]+1+len[x]/2-len[trans[x]]);
        ans=min(ans,n-len[x]+f[x]);
        for(int i=1;i<=4;i++)
        {
            if(!ch[x][i])continue;
            int y=ch[x][i];
            f[y]=min(f[y],f[x]+1);
            q.push(y);
        }
    }
}
int main()
{
    scanf("%d",&T);
    while(T--)
    {
        scanf("%s",s+1);n=strlen(s+1);
        s[0]='#';
        init();
        for(int i=1;i<=n;i++)add(change(s[i]),i);
        ans=n;solve();
        printf("%d\n",ans);
    }
    return 0;
}

原文地址:https://www.cnblogs.com/nofind/p/12070004.html

时间: 2024-10-11 04:07:10

P4762 [CERC2014]Virus synthesis的相关文章

luogu P4762 [CERC2014]Virus synthesis (回文自动机)

大意: 初始有一个空串, 操作(1)在开头或末尾添加一个字符. 操作(2)在开头或末尾添加该串的逆串. 求得到串$S$所需最少操作数. 显然最后一定是由某个偶回文通过添加字符得到的, 那么只需要求出所有偶回文的最少操作数即可. 结论: 偶回文最后一次进行翻倍操作一定最优. 证明考虑数学归纳, 对于长为$2$的回文串显然成立. 对长度$>2$的偶回文串$A$, 记最后一次翻倍得到的串$B$, $B$的一半为$C$. 记$f(S)$为串$S$的最优值, 就有$f(B)=f(C)+1$. 考虑由$B$

BZOJ4044 : [Cerc2014] Virus synthesis

设f[x]表示得到x这个回文串的最小步数,则ans=min(n-len[x]+f[x]) 边界条件f[长度为0的偶回文串]=1 因为翻转只会得到偶回文串,所以f[奇回文串]=该串的长度 对于一个偶回文串x,设y为x去掉首尾得到的串,有f[x]=f[y]+1 设y为长度不超过x的一半的x的最长回文后缀,有f[x]=len[x]/2-len[y]+f[y]+1 两种情况取个最小值即可. 对于状态的表示以及最长回文后缀的询问,用回文树支持操作即可.时间复杂度$O(n)$. #include<cstdi

bzoj4044/luoguP4762 [Cerc2014]Virus synthesis(回文自动机+dp)

bzoj Luogu 你要用ATGC四个字母用两种操作拼出给定的串: 1.将其中一个字符放在已有串开头或者结尾. 2.将已有串复制,然后reverse,再接在已有串的头部或者尾部. 一开始已有串为空.求最少操作次数. len<=100000 题解时间 一个非空串经过一次操作2之后一定是一个偶回文串. 所以考虑建出PAM之后dp. 由于只有偶回文串可以通过操作2产生,所以只有偶回文串参与dp. 对于PAM上一个点 $ x $ : 如果这个串去掉两头一个字母之后的串 $ y $ 非空,那么很明显 $

[BZOJ4044]Virus synthesis 回文自动机的DP

4044: [Cerc2014] Virus synthesis Time Limit: 20 Sec  Memory Limit: 128 MB Description Viruses are usually bad for your health. How about fighting them with... other viruses? In this problem, you need to find out how to synthesize such good viruses. W

Bzoj4044 Virus synthesis

题意 你要用 \(ATGC\) 四个字母用两种操作拼出给定的串: 将其中一个字符放在已有串开头或者结尾 将已有串复制,然后 \(reverse\) ,再接在已有串的头部或者尾部 一开始已有串为空.求最少操作次数. \(len\le100000\) Sol 首先有个结论 每次形成偶数长度回文串的最后一步一定是操作 \(2\) 那么考虑一个 \(DP\) 设 \(f[i]\) 表示形成 \(i\) 表示的字符串需要的最少步数 可以去掉首和尾转移来,可以由它的一个前缀或者后缀转移来 如果是个偶数长度的

G - Virus synthesis(回文树+dp)

题意 有AGTC四种字符,一开始有一个空串,每次操作,可以在首或尾加任意个字符,或者将已有字符镜面复制(左右两种复制方法),要求最少的操作步数使得得到给出的字符串 Solution: 我们要知道我们要求的是什么,是每一给回文串的所需要的最少的构造操作加上n-这个串的长度 然后就考虑如何求前者 题解:https://blog.csdn.net/nudt_spy/article/details/100061078 Code; #include<bits/stdc++.h> #define N 10

最近碰到了一个病毒木马:virus.win32.ramnit.B

由于 使用了 简单游 平台上的挂机工具: 番茄-自动人机对战免费版1217  ,使用了很久,头段时间家里电脑 360提示有病毒,本来我一直忽略的,但 我扫描了一下,大量的这个木马,于是 吧 简单游卸载了,全盘扫描,杀毒,修复了许多文件.许多文件被感染,文件大小都变了,还好 360 可以修复感染文件. 这一次,我是用公司电脑,再次 尝试使用 简单游 的挂机工具,没想到 依旧 是这个病毒 virus.win32.ramnit.B ,又是感染了大量的文件.360 居然没有提醒,还好 我清理电脑垃圾 顺

【Paper Reading】Improved Textured Networks: Maximizing quality and diversity in Feed-Forward Stylization and Texture Synthesis

Improved Textured Networks: Maximizing quality and diversity in Feed-Forward Stylization and Texture Synthesis https://arxiv.org/abs/1701.02096v1 本文最主要的贡献有两点: 1. 引入instance normalization 代替 batch normalization 2. 通过使得生产器从Julesz ensemble无偏采样来增加texture

zoj 3430 Detect the Virus(AC自动机)

题目连接:zoj 3430 Detect the Virus 题目大意:给定一个编码完的串,将每一个字符对应着表的数值转换成6位二进制,然后以8为一个数值,重新形成字符 串,判断给定询问串是否含有字符集中的串. 解题思路:主要是题意,逆编码部分注意,转换完了之后,可能有字符'\0',所以不能用字符串的形式储存,要用int型 的数组.注意有相同串的可能. #include <cstdio> #include <cstring> #include <queue> #incl