Codeforces 506E Mr. Kitayuta's Gift (矩阵乘法,动态规划)
描述:
给出一个单词,在单词中插入若干字符使其为回文串,求回文串的个数(|s|<=200,n<=10^9)
这道题超神奇,不可多得的一道好题
首先可以搞出一个dp[l][r][i]表示回文串左边i位匹配到第l位,右边i位匹配到第r位的状态数,可以发现可以用矩阵乘法优化(某人说看到n这么大就一定是矩阵乘法了= =)
但这样一共有|s|^2个节点,时间复杂度无法承受
我们先把状态树画出来:例如add
可以发现是个DAG
我们考虑把单独的每条链拿出来求解,那么最多会有|s|条不同的链,链长最多为|s|,时间复杂度为O(|s|^4log n)还是得跪
好像没什么思路了对吧= =(我第一步转化就没想到了= =)
我们考虑记有24个自环的为n24,25个自环的为n25,可以发现n24+n25*2=|s|或|s|+1也就是说对于一个确定的n24,一定有一个确定的n25
那么这样构图:
可以发现所有状况都被包括进来了!!!
那么一共有2|s|个节点,时间复杂度降了一个|s|,看上去好像还是不行
压常数= =
可以发现这个是棵树,也就是说如果按拓扑序编号的话,到时的矩阵左下角将是什么都没有的
那么就直接for i = 1 to n j = i to n k=i to j 就行了 = =
总结下吧
这道题为何神奇呢
首先它把一个DAG的图拆成了若干条相似的链
然后它又把这些链和成了一个更和谐的图
最后再观察题目性质得到一个比较神奇的优化方法
这给了我们什么启迪呢= =
首先遇到某些DAG我们可以考虑拆成若干条相似的链
遇到某些链我们可以考虑把他们合成一个图
最重要的是,还是得参透题目的性质
这道题基本都是依靠题目的性质到达下一步的,只有真正读懂读透这道题,我们才能想出更好的解法
CODE:
#include<cstdio>
#include<iostream>
#include<cstring>
#include<algorithm>
using namespace std;
#define maxn 410
#define mod 10007
typedef int ll;
struct marix{
int r,c;ll a[maxn][maxn];
inline void init(int x){r=c=x;for (int i=;i<=x;i++) a[i][i]=;}
}x,y;
inline void muti(marix &ans,const marix x,const marix y){
ans.r=ans.c=x.r;
for (int i=;i<=x.r;i++)
for (int j=i;j<=y.c;j++) {
int tmp=;
for (int k=i;k<=j;k++)
(tmp+=x.a[i][k]*y.a[k][j])%=mod;
ans.a[i][j]=tmp;
}
}
inline void power(marix &ans,marix x,int y) {
ans.init(x.r);
for (;y;y>>=) {
if (y&) muti(ans,ans,x);
muti(x,x,x);
}
}
ll f[][][];
char s[maxn];
inline ll calc(int l,int r,int x) {
ll &u=f[x][l][r];
if (u!=-) return u;
u=;
if (l==r) return u=x==;
if (s[l]==s[r]) {
if (l+==r) return u=x==;
return u=calc(l+,r-,x);
}
if (x>) return u=(calc(l+,r,x-)+calc(l,r-,x-))%mod;
return u;
}
int main(){
int n,m;
memset(f,-,sizeof(f));
scanf("%s",s+);
scanf("%d",&n);
m=strlen(s+);
n+=m;
int l=(n+)/,n24=m-,n25=(m+)/,n26=n25;
x.r=x.c=n24+n25+n26;
for (int i=;i<=n24;i++) x.a[i][i]=,x.a[i][i+]=;
for (int i=n24+;i<=n25+n24;i++) x.a[i][i]=,x.a[i][i+n25]=;
for (int i=n24+n25+;i<=n25+n24+n26;i++) x.a[i][i]=;
for (int i=n24+;i<n25+n24;i++) x.a[i][i+]=;
marix y;
power(y,x,l-);
muti(x,y,x);
ll ans;
for (int i=;i<=n24;i++) {
int j=(m-i+)/,k=l-i-j;
if (k<) continue;
ll sum=calc(,m,i);
(ans+=sum*x.a[n24-i+][n24+j+n25]%mod)%=mod;
if ((n&)&&(m-i&^))
(ans=ans-sum*y.a[n24-i+][n24+j]%mod+mod)%=mod;
}
printf("%d\n",ans);
return ;
}
Codeforces 506E Mr. Kitayuta's Gift (矩阵乘法,动态规划)的更多相关文章
- Codeforces 506E - Mr. Kitayuta's Gift(神仙矩阵乘法)
Codeforces 题目传送门 & 洛谷题目传送门 神仙题 %%%%%%%%%%%%% u1s1 感觉这道题风格很省选( 下记 \(m=|s|\),首先探讨 \(n+m\) 为偶数的情形. ...
- Codeforces 505A Mr. Kitayuta's Gift 暴力
A. Mr. Kitayuta's Gift time limit per test 1 second memory limit per test 256 megabytes input standa ...
- codeforces 505A. Mr. Kitayuta's Gift 解题报告
题目链接:http://codeforces.com/problemset/problem/505/A 题目意思:给出一个长度不大于10的小写英文字符串 s,问是否能通过在字符串的某个位置插入一个字母 ...
- 【CF506E】Mr. Kitayuta's Gift dp转有限状态自动机+矩阵乘法
[CF506E]Mr. Kitayuta's Gift 题意:给你一个字符串s,你需要在s中插入n个字符(小写字母),每个字符可以被插在任意位置.问可以得到多少种本质不同的字符串,使得这个串是回文的. ...
- 水题 Codeforces Round #286 (Div. 2) A Mr. Kitayuta's Gift
题目传送门 /* 水题:vector容器实现插入操作,暴力进行判断是否为回文串 */ #include <cstdio> #include <iostream> #includ ...
- codeforces Round 286# problem A. Mr. Kitayuta's Gift
Mr. Kitayuta has kindly given you a string s consisting of lowercase English letters. You are asked ...
- CodeForces 505B Mr. Kitayuta's Colorful Graph
Mr. Kitayuta's Colorful Graph Time Limit:1000MS Memory Limit:262144KB 64bit IO Format:%I64d ...
- codeforces 505B Mr. Kitayuta's Colorful Graph(水题)
转载请注明出处: http://www.cnblogs.com/fraud/ ——by fraud Mr. Kitayuta's Colorful Graph Mr. Kitayut ...
- [Codeforces 505C]Mr. Kitayuta, the Treasure Hunter
Description The Shuseki Islands are an archipelago of 30001 small islands in the Yutampo Sea. The is ...
随机推荐
- 长平狐 Cocos2d-x 的“HelloWorld” 深入分析
Cocos2d-x 的“HelloWorld” 深入分析 本节所用Cocos2d-x版本:cocos2d-1.0.1-x-0.12.0 不能免俗,一 ...
- oracle系列--级联删除和级联更新
必须声明:此博客转载于Oracle外键级联删除和级联更新http://www.2cto.com/database/201507/417496.html 鉴于此前收藏的精彩博客无料被删除了,很是痛心,所 ...
- Python dir()函数
您可以使用内置的dir()函数列出一个定义对象的标识符.例如,对于一个模块,包括在模块中定义的函数,类和变量. 当你给dir()提供一个模块名字时,它返回在那个模块中定义的名字的列表.当没有为其提供参 ...
- 【单调栈】hdu1506 Largest Rectangle in a Histogram
单调栈的介绍及一些基本性质 http://blog.csdn.net/liujian20150808/article/details/50752861 依次把矩形塞进单调栈,保持其单增,矩形中的元素是 ...
- PHP 中 static 和 self 的区别
使用 self:: 或者 __CLASS__ 对当前类的静态引用,取决于定义当前方法所在的类: 使用 static:: 不再被解析为定义当前方法所在的类,而是在实际运行时计算的.也可以称之为" ...
- div+CSS实现段落首行缩进两个字符
段落前面空两个字的距离,不要再使用空格了,用CSS实现段落首缩进两个字符.应该使用首行缩进text-indent.text-indent可以使得容器内首行缩进一定单位.比如中文段落一般每段前空两个汉字 ...
- 网上搜集的一段php可逆加密函数
php加密函数: function my_encrypt($data, $key='unun.in') { $char = $str = ''; $key = md5($key); $x = 0; $ ...
- MySQL要导出成excel的方法
MySQL 要导出成 excel 文件很简单,执行类似这样的命令: select * from 某个表 into outfile 'd:/文件名.xls'; 上述命令你在服务器上执行,就导在 ...
- TortoiseSVN使用简介
TortoiseSVN使用简介 2009-04-24 来源: dev.idv.tw 1.安装及下载client 端 2.什么是SVN(Subversion)? 3.为甚么要用SVN? 4.怎么样在Wi ...
- Java Swing jpanel paint方法执行两次的问题
Java Swing jpanel paint方法执行两次的问题: 在其他环境下执行了两次,自己测试怎么都是执行了一次,记录一下这个问题:需要后继工作: 可能是进行各种参数设置的时候导致了paint方 ...