【GDOI2016模拟3.15】基因合成(回文串+性质+DP)
【GDOI2016模拟3.15】基因合成
题意:
- 给一个目标串,要求从空串进行最少的操作次数变成目标串,操作有两种:
- 在串的头或尾加入一个字符.
- 把串复制一遍后反向接到串的末尾.
因为有回文操作,所以可以想到一些与回文有关的东西.
如Manacher,回文树……
这里采用强大的回文树.
首先注意到目标串可以看作是由一个长度为偶数的回文串在首尾加上若干字符得到的一个串.
所以我们可以求出原串中所有的偶回文串,然后再加加减减一下.
但为什么长度为奇数的不用讨论呢?
这是因为长度为奇数的回文串不可能通过第二个操作得到,所以不管如何,它对答案都没有贡献,它只能通过一个字符一个字符的加得到,尽管它的子串里可能有长度为偶数的回文串,但长度为偶数的回文串无论如何不可能通过第二个操作变为长度为奇数的回文串,所以我们依然可视作它是一个字符一个字符加入的.
然后我们继续讨论长度为偶数的回文串.
首先它可以由它的父亲状态得来,也就是在它父亲状态的最少操作数上+1得到.
当然,它也可以由一个不超过其自身长度一般的最长回文后缀进行一些操作后得来.
两者取最小值即为此偶回文串的最少操作数.
这实质上就是个在回文树上DP的东西.
现在问题还存在的是如何找到一个不超过其自身长度一半的偶回文串.
这一点我们可以借鉴回文树求\(fail\)的过程,我们设一个\(trans\)表示类似的东西,只不过多一个限制就是不能超过其长度一半.
然后每次从一个状态的父亲状态的\(trans\)开始找起来.
同样可以势能分析,时间复杂度显然是\(O(n)\)的.
当然,还可以打倍增....不过比较蛋疼。。。
#include <cstdio>
#include <cstring>
#include <iostream>
#define I register int
#define F(i, a, b) for (I i = a; i <= b; i ++)
#define mem(a, b) memset(a, b, sizeof a)
#define mn(a, b) ((a) = (a) < (b) ? (a) : (b))
const int N = 1e5 + 10, M = 26;
using namespace std;
int T; char ch[N];
int cur, lens, las, p, now, ans, len[N], fail[N], son[N][M], f[N], trans[N];
int Node(I x) { len[p] = x; return p ++; }
int getfail(I x) { while (ch[now] ^ ch[now - len[x] - 1]) x = fail[x]; return x; }
void Add(I x) {
cur = getfail(las);
if (!son[cur][x]) {
I t = Node(len[cur] + 2);
fail[t] = son[getfail(fail[cur])][x];
son[cur][x] = t;
}
las = son[cur][x];
if (len[las] & 1)
f[las] = len[las];
else
{
f[las] = f[cur] + 1;
I k = trans[cur];
while (ch[now - len[k] - 1] ^ ch[now] || (len[k] + 2) * 2 > len[las]) k = fail[k];
trans[las] = son[k][x];
mn(f[las], f[trans[las]] + len[las] / 2 - len[trans[las]] + 1);
}
mn(ans, f[las] + lens - len[las]);
}
int main() {
for (scanf("%d", &T); T --; ) {
scanf("%s", ch + 1);
p = 0, lens = strlen(ch + 1);
Node(0), Node(- 1), f[0] = 1;
fail[0] = 1, las = 0, ans = 1e9;
F(i, 1, lens) now = i, Add(ch[i] - 'A');
printf("%d\n", ans);
F(i, 0, p + 2) mem(son[i], 0);
}
}
【GDOI2016模拟3.15】基因合成(回文串+性质+DP)的更多相关文章
- 关于回文串的DP问题
问题1:插入/删除字符使得原字符串变成一个回文串且代价最小 poj 3280 Cheapest Palindrome 题意:给出一个由m中字母组成的长度为n的串,给出m种字母添加和删除花费的代价,求让 ...
- 回文串 --- 动态dp UVA 11584
题目链接: https://cn.vjudge.net/problem/34398/origin 本题的大意其实很简单,就是找回文串,大致的思路如下: 1. 确定一个回文串,这里用到了自定义的chec ...
- poj3280 Cheapest Palindrome(回文串区间dp)
https://vjudge.net/problem/POJ-3280 猛刷简单dp第一天第三题. 这个据说是[求字符串通过增减操作变成回文串的最小改动次数]的变体. 首先增减操作的实质是一样的,所以 ...
- 1154 回文串划分(DP+Manacher)
1154 回文串划分 基准时间限制:1 秒 空间限制:131072 KB 分值: 40 难度:4级算法题 有一个字符串S,求S最少可以被划分为多少个回文串. 例如:abbaabaa,有多种划分方式. ...
- uva 10453 【回文串区间dp】
Uva 10453 题意:给定字符串,问最少插入多少个字符使其变成回文串,并任意输出一种结果. 题解:和Uva 10739类似,这里是只能增加.类似定义dp[i][j]表示子串Si...Sj变为回文串 ...
- [LeetCode] Palindrome Partitioning 拆分回文串
Given a string s, partition s such that every substring of the partition is a palindrome. Return all ...
- 【Manacher算法】最长子回文串
[Manacher算法] 这个算法用来找出一个字符串中最长的回文子字符串. 如果采取暴力解最长回文子字符串问题,大概可以有两种思路:1. 遍历出所有子字符串找其中最长的回文 2. 从每个字符作为中心, ...
- 计蒜之道 初赛 第三场 题解 Manacher o(n)求最长公共回文串 线段树
腾讯手机地图 腾讯手机地图的定位功能用到了用户手机的多种信号,这当中有的信号的作用范围近.有的信号作用的范围则远一些.有的信号相对于用户在不同的方位强度是不同的,有的则是在不论什么一个方向上信号强度都 ...
- 马拉车,O(n)求回文串
马拉车,O(n)求回文串 对整个马拉车算法步骤做个总结: 第一步:将每个原字母用两个特殊字符包围如: aaa --> #a#a#a# abab -->#a#b#a#b 同时可以由这个翻倍的 ...
随机推荐
- Spring 事务与脏读、不可重复读、幻读
索引: 目录索引 参看代码 GitHub: 1.Spring 事务 2.事务行为 一.Spring 事务: Spring 的事务机制是用统一的机制来处理不同数据访问技术的事务处理. Spring 的事 ...
- Python面试笔记二
一.算法 1.归并排序 2.快速排序 3.算法复杂度 4.哈希表数据结构 二.数据库 1.设计一个用户关注系统的数据库表 1.设计一个用户关注系统的数据库表,写三个相关的SQL语句两张表,一张user ...
- 面向对象_new,__eq__,__hash__
老师博客:http://www.cnblogs.com/Eva-J/articles/7351812.html __new__ __init__是一种初始化的方法 __new__是构建方法,创建一个对 ...
- 解决Error response from daemon: Get https://registry-1.docker.io/v2/library/hello-world/manifests/
https://blog.csdn.net/quanqxj/article/details/79479943
- supervisor 工具 配置
配置supervisor工具,管理django后台 supervisor管理进程,是通过fork/exec的方式将这些被管理的进程当作supervisor的子进程来启动,所以我们只需要将要管理进程的可 ...
- SpringBoot四大核心
auto-configuration.starters.cli.actuator
- 将docker镜像上传到docker hub
- SpringBoot 数据篇之使用JDBC
SpringBootTutorial :: Data :: Jdbc 简介 API execute update query 实战 配置数据源 完整示例 引申和引用 简介 Spring Data 包含 ...
- TypeError: argument to reversed() must be a sequence ERROR basehttp 124 "GET /admin/ HTTP/1.1" 500 114103 Performing system checks...
Error Msg TypeError: argument to reversed() must be a sequence ERROR basehttp 124 "GET /admin/ ...
- PS绘制飘逸彩色丝带教程
一.新建一个大小适当的图像,点击工具栏上的钢笔工具,使用形状图层来绘制出下图的形状. 二.把形状所在层的填充设为0%,填充设成0是不会影响到图层的,不像不透明度那样会影响图层样式的效果. 三.双击丝带 ...