bzoj 1138: [POI2009]Baj 最短回文路 dp优化
1138: [POI2009]Baj 最短回文路
Time Limit: 10 Sec Memory Limit: 162 MB
Submit: 161 Solved: 48
[Submit][Status]
Description
N个点用M条有向边连接,每条边标有一个小写字母。 对于一个长度为D的顶点序列,回答每对相邻顶点Si到Si+1的最短回文路径。 如果没有,输出-1。 如果有,输出最短长度以及这个字符串。
Input
第一行正整数N和M ( 2 ≤ N ≤ 400 , 1 ≤ M ≤ 60,000 ) 接下来M行描述边的起点,终点,字母。接下来D表示询问序列长度 ( 2 ≤ D ≤ 100 ) 再接下来D个1到N的整数
Output
对于D-1对相邻点,按要求输出一行。如果没合法方案,输出-1。 如果有合法,输出最短长度
Sample Input
1 2 a
1 3 x
1 4 b
2 6 l
3 5 y
4 5 z
6 5 a
3
1 5 3
Sample Output
-1
一看题,这不是水dp么?设dp[i][j]为i到j的回文最短路长度,随便转移一下就行了,写出来才发现TLE的一塌糊涂。在一系列常数优化后直接在网上找题解了。。。
优化0:这算不上什么优化吧,由于每次只枚举特定起点特定颜色的边,用前向星比枚举边要快一些。
优化1:这个for是不是嵌套的太多了,冗余状态很多,用类似于spfa的方法优化一下。
优化2:直觉时卡时间的点应该有很多重边,先将边去重。
优化3:是不是觉得我们在dp[i][j]向下转移的过程中分别从起点终点枚举边很慢,我们外加一个数组dp2[i][j][k]表示从i到j经过一条边权为k的边和一个回文串的最短距离,每次转移变为了dp与dp2的相互更新,很不幸的一点是仔细想一想这样并不能优化时间复杂度级别,但是是一个很强劲的剪枝。
网上的题解称光优化3就能过,本人太弱,用了优化3还是T,就把优化1,2,3都加上了。
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
#define MAXN 450
#define MAXM 61000
#define INF 0x3f3f3f3f
struct edge
{
int s,t,c;
}e[MAXM],e2[MAXM];
bool cmp_edge(edge e1,edge e2)
{
if (e1.s!=e2.s)return e1.s<e2.s;
if (e1.c!=e2.c)return e1.c<e2.c;
if (e1.t!=e2.t)return e1.t<e2.t;
return false;
}
bool cmp_edge2(edge e1,edge e2)
{
if (e1.t!=e2.t)return e1.t<e2.t;
if (e1.c!=e2.c)return e1.c<e2.c;
if (e1.s!=e2.s)return e1.s<e2.s;
return false;
}
bool operator ==(edge e1,edge e2)
{
return e1.s==e2.s && e1.t==e2.t && e1.c==e2.c;
}
int head[MAXN*],head2[MAXN*];
int dp[MAXN][MAXN],dp2[MAXN][MAXN][];
inline bool deal(int &x,int y)
{
if (x>y)
{
x=y;
return true;
}else
return false;
}
pair<int,int> q[];
bool vis[MAXN][MAXN];
int main()
{
freopen("input.txt","r",stdin);
int i,j,k,x,y,z,n,m,a,b;
scanf("%d%d\n",&n,&m);
char ch;
for (i=;i<m;i++)
{
scanf("%d%d %c\n",&x,&y,&ch);
e[i].s=x;
e[i].t=y;
e[i].c=ch-'a';
e2[i]=e[i];
}
sort(e,e+m,cmp_edge);
sort(e2,e2+m,cmp_edge2);
x=m;
m=unique(e,e+x)-e;
m=unique(e2,e2+x)-e2;
for (i=;i<m;i++)
{
head[e[i].s*+e[i].c]++;
head2[e[i].t*+e[i].c]++;
}
for (i=;i<=n*+;i++)
head[i]+=head[i-],head2[i]+=head2[i-];
for (i=n*+;i>=;i--)
head[i]=head[i-],head2[i]=head2[i-];
head[]=;head2[]=;
e[m].s=e2[m].t=INF; memset(dp,INF,sizeof(dp));
memset(dp2,INF,sizeof(dp2));
for (i=;i<=n;i++)
dp[i][i]=;
for (i=;i<m;i++)
dp[e[i].s][e[i].t]=;
int h=-,t=-;
for (i=;i<=n;i++)
for (j=;j<=n;j++)
if (dp[i][j]!=INF)
q[++t]=make_pair(i,j),vis[i][j]=true;; while (h<t)
{
x=q[++h].first;
y=q[h].second;
vis[x][y]=false;
for (a=head2[x*+k];e2[a].t==x;a++)
{
if (dp2[e2[a].s][y][e2[a].c]>dp[x][y]+)
{
dp2[e2[a].s][y][e2[a].c]=dp[x][y]+;
if (!vis[e2[a].s][y])
{
vis[e2[a].s][y]=true;
q[++t]=make_pair(e2[a].s,y);
}
}
}
for (b=head[y*+k];e[b].s==y;b++)
{
if (dp[x][e[b].t]>dp2[x][y][e[b].c]+)
{
dp[x][e[b].t]=dp2[x][y][e[b].c]+;
if (!vis[x][e[b].t])
{
vis[x][e[b].t]=true;
q[++t]=make_pair(x,e[b].t);
}
}
}
}
int q;
scanf("%d",&q);
scanf("%d",&x);
for (i=;i<q;i++)
{
scanf("%d",&y);
if (dp[x][y]==INF)
{
printf("%d\n",-);
}else
{
printf("%d\n",dp[x][y]);
}
x=y;
}
}
bzoj 1138: [POI2009]Baj 最短回文路 dp优化的更多相关文章
- bzoj 1138: [POI2009]Baj 最短回文路
额,,貌似网上的题解都说超时之类的. 然而我这个辣鸡在做的时候不知道在想什么,连超时的都不会. 超时的大概是这样的,f[x][y]表示x到y的最短回文路,然后更新的话就是 f[x][y]更新到 f[a ...
- [BZOJ1138][POI2009]Baj 最短回文路
[BZOJ1138][POI2009]Baj 最短回文路 试题描述 N个点用M条有向边连接,每条边标有一个小写字母. 对于一个长度为D的顶点序列,回答每对相邻顶点Si到Si+1的最短回文路径. 如果没 ...
- [Swift]LeetCode214. 最短回文串 | Shortest Palindrome
Given a string s, you are allowed to convert it to a palindrome by adding characters in front of it. ...
- [python,2019-02-15] 最短回文串
给定一个字符串 s,你可以通过在字符串前面添加字符将其转换为回文串.找到并返回可以用这种方式转换的最短回文串. 示例 1: 输入: "aacecaaa" 输出: "aaa ...
- leetcode 214. 最短回文串 解题报告
给定一个字符串 s,你可以通过在字符串前面添加字符将其转换为回文串.找到并返回可以用这种方式转换的最短回文串. 示例 1: 输入: "aacecaaa" 输出: "aaa ...
- 214 Shortest Palindrome 最短回文串
给一个字符串 S, 你可以通过在字符串前面添加字符将其转换为回文串.找到并返回可以用这种方式转换的最短回文串.例如:给出 "aacecaaa",返回 "aaacecaaa ...
- Leetcode 214.最短回文串
最短回文串 给定一个字符串 s,你可以通过在字符串前面添加字符将其转换为回文串.找到并返回可以用这种方式转换的最短回文串. 示例 1: 输入: "aacecaaa" 输出: &qu ...
- Java实现 LeetCode 214 最短回文串
214. 最短回文串 给定一个字符串 s,你可以通过在字符串前面添加字符将其转换为回文串.找到并返回可以用这种方式转换的最短回文串. 示例 1: 输入: "aacecaaa" 输出 ...
- bzoj2084/luoguP3501 [Poi2010]Antisymmetry(回文自动机+dp)
bzoj2084/luoguP3501 [Poi2010]Antisymmetry(回文自动机+dp) bzoj Luogu 对于一个01字符串,如果将这个字符串0和1取反后,再将整个串反过来和原串一 ...
随机推荐
- 读写应用程序数据-SQLite3
SQLite3是嵌入到ios中的关系型数据库.对存储大规模的数据非常实用,使得不必将每个对象加到内存中. 支持NULL.INTEGER.REAL(浮点数字).TEXT(字符串和文本).BLOB(二进制 ...
- (原创)speex与wav格式音频文件的互相转换
我们的司信项目又有了新的需求,就是要做会议室.然而需求却很纠结,要继续按照原来发语音消息那样的形式来实现这个会议的功能,还要实现语音播放的计时,暂停,语音的拼接,还要绘制频谱图等等. 如果是wav,m ...
- iOS自定义UICollectionViewLayout之瀑布流
目标效果 因为系统给我们提供的 UICollectionViewFlowLayout 布局类不能实现瀑布流的效果,如果我们想实现 瀑布流 的效果,需要自定义一个 UICollectionViewLay ...
- 一种通用数据采集的schema定义形式
{ "name": "凤凰金融", "notice": { "data": "attribute", ...
- redis resque消息队列
Resque 目前正在学习使用resque .resque-scheduler来发布异步任务和定时任务,为了方便以后查阅,所以记录一下. resque和resque-scheduler其优点在于功能比 ...
- C#如何配置应用程序域
转载:http://www.csharpwin.com/csharpspace/9175r9023.shtml 您可以使用 AppDomainSetup 类,为新应用程序域提供带有配置信息的公共语言运 ...
- ios Objective-C的动态特性
这是一篇译文,原文在此,上一篇文章就是受这篇文章启发,这次干脆都翻译过来. 过去的几年中涌现了大量的Objective-C开发者.有些是从动态语言转过来的,比如Ruby或Python,有些是从强类型语 ...
- 一些简单的帮助类(1)-- String的类型验证
在工作中经常会遇到 验证String 中的值是否属于Int型或者是Bool又或是Date 一般的做法是用方法 类型.TryParse(string,类型) 来做验证. "; int intV ...
- Java教程——CMD手动编译运行失败原因(高手略过)
(仅对新手,高手略过)在学习Java初期,我们在利用cmd手动编译java程序的时候,会遇到编译成功,但运行却总是提示失败.已经排除了java配置环境的问题,Path和ClassPath以及%JAVA ...
- 项目报错-无法解析类型 XXXX.xx 从必需的 .class 文件间接引用了它
这个编译错误有几个原因 1.jdk版本问题 要是报错是某些java包里的东西那就可以肯定是jdk版本的问题, 比如无法解析类型java.lang.Object,无法解析类型java.lang.Char ...