好难啊。。根本不会做。。基本上是抄Claris。。。

题目链接: (bzoj)https://www.lydsy.com/JudgeOnline/problem.php?id=4044

(luogu)https://www.luogu.org/problemnew/show/P4762

题解: 先观察到三个(ju)性(fei)质(hua): 2操作的结果一定是一个回文串(废话), 最后一个2操作之后只能进行1操作(废话), 只进行1操作花费代价等于字符个数(废话)

三句废话连在一起说: 最后一次2操作产生了一个回文串,此后只能进行1操作,总代价等于回文串代价+目标串长度-回文串长度

又因为一个串的本质不同回文子串个数是\(O(n)\)个,而且奇数长度的回文串没有任何意义,所以实际上有用的状态只是这个串的那些偶数长度的回文子串。。。

然后就可以在回文自动机上DP

考虑如何生成一个偶数长度的回文串,最优方案最后一步肯定是2操作(废话),那么考虑最后2操作之前的一步

因为我们只记录回文串的状态,所以我们希望建立从回文串到回文串的转移,而不能依赖于其他子串。

第一种情况,2操作之前的最后一步将一个字符加在了外面,则代价为该串去掉两头字符后的回文串代价+1 (不需要考虑更多,因为回文串去掉两头还是回文串)

第二种情况,2操作之前的最后一步将字符加在了里面,于是只能找到该串右半侧的最长回文后缀(或左半侧的最长回文前缀)与之建立联系,则代价为(该串串长的一半-右半侧最长回文后缀长度)+右半侧最长回文后缀代价+1 (最后一个+1是指要复制一次,但是第一种情况的+1指的是在复制之前加了一个字符,最后一步复制的代价在去掉两头字符的代价中算过了)

要注意初始值是\(f[0]=1\), 因为要求的是长度为0的回文串的代价,相当于自我复制一遍,以供第一种情况转移。

最后的问题是,如何对于一个串的一个回文子串快速求出不超过其长度一半的最长回文后缀?

建回文自动机,在上面进行递推

一开始觉得是从它的fail递推下来十分直观(因为fail本来就是最长回文后缀),但是实际上这样并不好(可能需要建出“fail树”再在上面做一些树的操作)。

比较好的方法应该是对于当前要递推的\(u\)点找到点\(p\)满足\(son[p][ch]=u\) (\(ch\)是\(u\)的结束字符,\(p\)点在构建回文自动机的过程中已经找到了),然后从\(p\)开始跳fail,直到合法为止,然后再走到它的\(ch\)儿子。因为这样可以避免在fail树上自上而下寻找,而采用更方便的\(son\)来向下寻找。

代码

#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<algorithm>
using namespace std; const int N = 1e5+2;
const int S = 4;
char a[N+3];
int len[N+3];
int fail[N+3];
int son[N+3][S+1];
int bd[N+3];
int dp[N+3];
int que[N+3];
int n,siz,lstpos; int decode(char ch)
{
if(ch=='A') return 1;
else if(ch=='C') return 2;
else if(ch=='T') return 3;
else if(ch=='G') return 4;
} void initPAM()
{
siz = 1; fail[0] = fail[1] = 1; len[0] = 0; len[1] = -1; lstpos = 1; bd[0] = 0; bd[1] = 1;
} void insertchar(int id)
{
// printf("insert %d\n",a[id]);
int p = lstpos;
while(a[id-1-len[p]]!=a[id]) {p = fail[p];}
if(!son[p][a[id]])
{
siz++; int u = siz,v = fail[p];
while(a[id-1-len[v]]!=a[id]) {v = fail[v];}
fail[u] = son[v][a[id]]; len[u] = len[p]+2; son[p][a[id]] = u;
// printf("p=%d u=%d\n",p,u);
if(len[u]<=2) {bd[u] = fail[u];}
else
{
bd[u] = bd[p];
while(a[id-1-len[bd[u]]]!=a[id] || (len[bd[u]]+2)*2>len[u])
{
bd[u] = fail[bd[u]];
}
bd[u] = son[bd[u]][a[id]];
}
}
lstpos = son[p][a[id]];
} void clear()
{
for(int i=0; i<=siz; i++) bd[i] = dp[i] = len[i] = fail[i] = son[i][1] = son[i][2] = son[i][3] = son[i][4] = 0;
} int main()
{
int T; scanf("%d",&T);
while(T--)
{
initPAM();
scanf("%s",a+1); n = strlen(a+1);
for(int i=1; i<=n; i++) a[i] = decode(a[i]);
for(int i=1; i<=n; i++)
{
insertchar(i);
}
// printf("siz=%d\n",siz);
// for(int i=0; i<=siz; i++) for(int j=1; j<=S; j++) {if(son[i][j]) printf("son%d %d %d\n",i,j,son[i][j]);}
// printf("fail: "); for(int i=0; i<=siz; i++) printf("%d ",fail[i]); puts("");
// printf("bd: "); for(int i=0; i<=siz; i++) printf("%d ",bd[i]); puts("");
for(int i=1; i<=siz; i++) {if(len[i]&1) dp[i] = len[i];}
int head = 1,tail = 1;
que[1] = 0; dp[0] = 1;
while(head<=tail)
{
int u = que[head]; head++;
for(int i=1; i<=S; i++)
{
if(son[u][i])
{
tail++; que[tail] = son[u][i];
dp[son[u][i]] = min(dp[u]+1,((len[son[u][i]])>>1)-len[bd[son[u][i]]]+dp[bd[son[u][i]]]+1);
}
}
}
int ans = n;
for(int i=0; i<=siz; i++) ans = min(ans,n-len[i]+dp[i]);
printf("%d\n",ans);
clear();
}
return 0;
}

BZOJ 4044 Luogu P4762 [CERC2014]Virus Synthesis (回文自动机、DP)的更多相关文章

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

    大意: 初始有一个空串, 操作(1)在开头或末尾添加一个字符. 操作(2)在开头或末尾添加该串的逆串. 求得到串$S$所需最少操作数. 显然最后一定是由某个偶回文通过添加字符得到的, 那么只需要求出所 ...

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

    bzoj4044/luoguP4762 [Cerc2014]Virus synthesis(回文自动机+dp) bzoj Luogu 你要用ATGC四个字母用两种操作拼出给定的串: 1.将其中一个字符 ...

  3. BZOJ 4044 Virus synthesis (回文自动机+dp)

    题目大意: 你可以在一个串的开头或者末尾加入一个字符,或者把当前整个串$reverse$,然后接在前面或者后面,求达到目标串需要的最少操作次数 对目标串建出$PAM$ 定义$dp[x]$表示当前在回文 ...

  4. bzoj 4044: Virus synthesis 回文自动机

    题目大意: 你要用ATGC四个字母用两种操作拼出给定的串: 将其中一个字符放在已有串开头或者结尾 将已有串复制,然后reverse,再接在已有串的头部或者尾部 一开始已有串为空.求最少操作次数. le ...

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

    4044: [Cerc2014] Virus synthesis Time Limit: 20 Sec  Memory Limit: 128 MB Description Viruses are us ...

  6. bzoj 4044 Virus synthesis - 回文自动机 - 动态规划

    题目传送门 需要高级权限的传送门 题目大意 要求用两种操作拼出一个长度为$n$的只包含'A','T','G','C'的字符串 在当前字符串头或字符串结尾添加一个字符 将当前字符串复制,将复制的串翻转, ...

  7. bzoj2084/luoguP3501 [Poi2010]Antisymmetry(回文自动机+dp)

    bzoj2084/luoguP3501 [Poi2010]Antisymmetry(回文自动机+dp) bzoj Luogu 对于一个01字符串,如果将这个字符串0和1取反后,再将整个串反过来和原串一 ...

  8. bzoj 4044: [Cerc2014] Virus synthesis【回文自动机+dp】

    建回文自动机,注意到一个回文串是可以通过一个长度小于等于这个串长度的一半的回文串添上一些字符然后复制得到的,也就是在自动机上向fa走,相当于treedp 每次都走显然会T,记录一个up,指向祖先中最下 ...

  9. 洛谷P4762 [CERC2014]Virus synthesis(回文自动机+dp)

    传送门 回文自动机的好题啊 先建一个回文自动机,然后记$dp[i]$表示转移到$i$节点代表的回文串的最少的需要次数 首先肯定2操作越多越好,经过2操作之后的串必定是一个回文串,所以最后的答案肯定是由 ...

随机推荐

  1. K8S之利用Label控制Pod位置

    首先介绍下什么是Label? Label是Kubernetes系列中一个核心概念.是一组绑定到K8s资源对象上的key/value对.同一个对象的labels属性的key必须唯一.label可以附加到 ...

  2. js 二叉搜索树

    二叉搜索树:顾名思义,树上每个节点最多只有二根分叉:而且左分叉节点的值 < 右分叉节点的值 . 特点:插入节点.找最大/最小节点.节点值排序 非常方便 1 2 3 4 5 6 7 8 9 10 ...

  3. 慕课网3-13编程练习:采用flex弹性布局制作页面主导航

    小伙伴们,伸缩容器的属性我们已经学完了,接下来使用我们所学的伸缩容器属性完成下面的效果图. 要求: 1.logo.导航项.登录注册按钮这三项在水平和垂直方向上都对齐,而且他们之间的距离也相等. 2.导 ...

  4. POJ 1873 计算几何

    思路: 二进制枚举一下要删哪些点 求个凸包,算一下贡献 //By SiriusRen #include <cmath> #include <cstdio> #include & ...

  5. BZOJ 3798 分块打表

    思路: 这题思路真是奇妙 先跑个暴力 每隔1e5打个表 块内暴力 打表程序: (开O3 15秒就跑完了) //By SiriusRen #include <bits/stdc++.h> u ...

  6. [Luogu 1966] noip13 火柴排队

    [Luogu 1966] noip13 火柴排队 Problem 涵涵有两盒火柴,每盒装有 n 根火柴,每根火柴都有一个高度. 现在将每盒中的火柴各自排成一列, 同一列火柴的高度互不相同, 两列火柴之 ...

  7. 涨知识III - 百度2016校园招聘——移动软件研发工程师

    1.列关于线程调度的叙述中,错误的是(). 正确答案 :BE A调用线程的sleep()方法,可以使比当前线程优先级低的线程获得运行机会 B调用线程的yeild()方法,只会使与当前线程相同优先级的线 ...

  8. 参加2016华为codecraft编程精英挑战赛后感

    2016年4月参加了华为的软件比赛. 关于比赛:给了一道图论的np-hard问题.刚开始完全不知道怎么入手,请教过师兄,自己也琢磨过,没有什么万全的解决方法.注意,这里说的是万全的办法.本科搞算法时候 ...

  9. 入门开发工具idea常见问题之选项中没有servlet

    1.在maven中如果创建不了servlet,在project Setting旁边的添加一个web选项,就可以创建servlet了. 初次接触这个陌生的工具还是不太好弄.

  10. JS——dom

    节点的获取 <script> var div = document.getElementById("box");//返回指定标签 var div = document. ...