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取反后,再将整个串反过来和原串一 ...
随机推荐
- Foundation: NSNotificationCenter
一个NSNotificationCenter对象(通知中心)提供了在程序中广播消息的机制,它实质上就是一个通知分发表.这个分发表负责维护为各个通知注册的观察者,并在通知到达时,去查找相应的观察者,将通 ...
- 如何运用管理员身份运行cmd窗口?
所有程序 → 附件 → 命令行提示符 → 鼠标右键“以管理员身份运行”.
- Java Inner class
2016-03-27 内部类:就是把一个类的定义放在另外一个外围类定义的里面. class OutterClass{ class InnerClass { } } 1. 内部类主要有四种:成员内部类( ...
- 运用json-lib生成特定json
在实现接口过程中,一般协议都是定义数据格式为json.我们有时候需要把bean转换为JSON输出给接口调用者,但是可能存在bean中的字段有些不是接口定义所需要的.这个时候需要我们对JSON转换是需要 ...
- FTP上传下载
使用的是apache开源包commons-net-3.3.jar所提供的FTPClient FTP服务器使用Quick Easy FTP Server 4.0.0(服务器ip为192.168.31.1 ...
- SQL Server 2008 Values 新用途
SQL Server 2008中新增功能:可以使用单个Insert命令插入多行. Create table Demo_Values (PKID int not null identity(1,1) p ...
- yield 关键字和迭代器
一般使用方法 yield 关键字向编译器指示它所在的方法是迭代器块 在迭代器块中,yield 关键字与 return 关键字结合使用,向枚举器对象提供值. 这是一个返回值,例如,在 forea ...
- Solr4.8.1与Tomcat7整合
Solr4.8.1和Tomcat7都可以到官方网站去下载,我这里就不多说了,如下图. 这里我们首先解压Solr-4.8.1.zip,再解压Tomcat,解压后,再在当前文件夹下建2个文件夹,一个用来放 ...
- java内存
java内存分为四部分: 1).栈区(stacksegment),由编译器自动分配释放,存放函数的参数值和局部变量的值等,具体方法执行结束之后,系统自动释放JVM内存资源: 2).堆区(heapseg ...
- 关于GC进行垃圾回收的时机
前言 今天查看一个同事的代码,发现代码中多处地方使用了GC.Collect()方法,我问他为什么这么做,他说感觉程序中定义了好多变量,怕GC回收不及时,用GC.Collect()可以手动掌控GC进行垃 ...