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取反后,再将整个串反过来和原串一 ...
随机推荐
- java class 文件解析
参考下面两个文章对一个class文件进行解析: http://docs.oracle.com/javase/specs/jvms/se7/html/jvms-4.html#jvms-4.4.6 htt ...
- ADO.NET连接数据库的两种方式
//实现了IDisposable接口的类,用using括起来 //插入数据 string connString = "Data Source=(local);Initial Catalog= ...
- sql 几个常用函数
ROUND 用法: --15.000 表示小数点第一位取四舍五入,将原小数点后的位数都设置为0SELECT ROUND(15.258,0) --15.300,第二个参数如果是1,则取原值小数点后第一位 ...
- Linq语法
希望能帮助一些linq新手. 开门见山 读这篇文章之前,我先说下,每一种搜索结果集,我都以三种方式变现出来,为啦更好的理解,希望不要嫌我啰嗦. 1.简单的linq语法 //1 var ss = fro ...
- EDMX更新实体后出现键值映射问题
近期做项目的EF改版时,在DB(ORACLE)中的表里添加一个新的PK,去除原有的PK. 在DB已添加完成操作,但这时在EDMX里进行从DB更新到EF里,更新完成后就发生如下错误提示: Error 6 ...
- Ext.Net学习笔记13:Ext.Net GridPanel Sorter用法
Ext.Net学习笔记13:Ext.Net GridPanel Sorter用法 这篇笔记将介绍如何使用Ext.Net GridPanel 中使用Sorter. 默认情况下,Ext.Net GridP ...
- height为100%的问题
问题描述 很多同学,对于设置div 的高度为100%时,有疑惑. 设置div 的高度为100%,意思是此 div 的高度 铺满父元素. 那么 怎么使 div 铺满浏览器屏幕? 代码 <!DOCT ...
- 今天收到报警邮件,提示网站502 bad gateway,
今天收到报警邮件,提示网站502 bad gateway, 输入网站url后果然无法打开: 登录服务器查看nginx进程正常: 查看fastcGI进程已经停止运行了: 问题找到后就该查找是什么原因产生 ...
- nginx 中出现nginx: [emerg] bind() to 0.0.0.0:80 failed (98: Address already in use)
有其他的程序占用的80端口.只需把相关程序关闭,fuser -k 80/tcp 然后再次 /usr/local/nginx/sbin/nginx,就能开启nginx服务了
- Linux文件权限学习总结
一.用户对文件或目录都有哪些权限? 四种:读.写.执行.没有权限 二.如何表示这四种权限? 如果用十进制数字表示,分别为:4.2.1.0:如果用字符表示,分别为:r.w.x.-.个人觉得,使用chmo ...