HDU4511 小明系列故事——女友的考验(AC自动机+DP)

题目大概说有平面有n个点,从1点出发走到n点,每一步只能走到序号比当前更大的点且走的序列不能包含给定的m个序列中的任何一个,问1走到n的最短路。

用m个序列建个AC自动机,后缀包含整个序列的结点标记一下,然后用dp[u][S]表示走到u点且走的序列的后缀状态是自动机上第S个结点的最短路,这样在AC自动机上跑着转移就OK了。

 1 #include<cstdio>
 2 #include<cstring>
 3 #include<cmath>
 4 #include<queue>
 5 #include<algorithm>
 6 using namespace std;
 7 #define MAXN 666
 8 int tn,ch[MAXN][55],fail[MAXN];
 9 bool flag[MAXN];
10 void insert(int *a,int n){
11     int x=0;
12     for(int i=0; i<n; ++i){
13         int y=a[i];
14         if(ch[x][y]==0) ch[x][y]=++tn;
15         x=ch[x][y];
16     }
17     flag[x]=1;
18 }
19 int n;
20 void getfail(){
21     memset(fail,0,sizeof(fail));
22     queue<int> que;
23     for(int i=1; i<=n; ++i){
24         if(ch[0][i]) que.push(ch[0][i]);
25     }
26     while(!que.empty()){
27         int x=que.front(); que.pop();
28         for(int i=1; i<=n; ++i){
29             if(ch[x][i]){
30                 que.push(ch[x][i]);
31                 fail[ch[x][i]]=ch[fail[x]][i];
32                 flag[ch[x][i]]|=flag[ch[fail[x]][i]];
33             }else ch[x][i]=ch[fail[x]][i];
34         }
35     }
36 }
37 double x[55],y[55],dist[55][55],d[55][MAXN];
38 int a[6];
39 int main(){
40     int m,k;
41     while(~scanf("%d%d",&n,&m) && (n||m)){
42         for(int i=1; i<=n; ++i){
43             scanf("%lf%lf",x+i,y+i);
44         }
45         tn=0;
46         memset(ch,0,sizeof(ch));
47         memset(flag,0,sizeof(flag));
48         while(m--){
49             scanf("%d",&k);
50             for(int i=0; i<k; ++i) scanf("%d",a+i);
51             insert(a,k);
52         }
53         getfail();
54         memset(dist,0,sizeof(dist));
55         for(int i=1; i<=n; ++i){
56             for(int j=i+1; j<=n; ++j){
57                 dist[i][j]=sqrt((x[i]-x[j])*(x[i]-x[j])+(y[i]-y[j])*(y[i]-y[j]));
58             }
59         }
60         for(int i=1; i<=n; ++i){
61             for(int j=0; j<=tn; ++j) d[i][j]=-1;
62         }
63         d[0][0]=0;
64         for(int i=0; i<n; ++i){
65             for(int j=0; j<=tn; ++j){
66                 if(d[i][j]<0) continue;
67                 if(i==0 && !flag[ch[0][1]]){
68                     d[1][ch[0][1]]=0;
69                     continue;
70                 }
71                 for(int k=i+1; k<=n; ++k){
72                     if(flag[ch[j][k]]) continue;
73                     if(d[k][ch[j][k]]<0 || d[k][ch[j][k]]>d[i][j]+dist[i][k]) d[k][ch[j][k]]=d[i][j]+dist[i][k];
74                 }
75             }
76         }
77         double res=-1;
78         for(int i=0; i<=tn; ++i){
79             if(d[n][i]<0) continue;
80             if(res<0 || res>d[n][i]) res=d[n][i];
81         }
82         if(res<0) puts("Can not be reached!");
83         else printf("%.2f\n",res);
84     }
85     return 0;
86 }
时间: 2024-11-16 08:42:38

HDU4511 小明系列故事——女友的考验(AC自动机+DP)的相关文章

HDU - 4511 小明系列故事――女友的考验(AC自动机+DP)

Description 终于放寒假了,小明要和女朋友一起去看电影.这天,女朋友想给小明一个考验,在小明正准备出发的时候,女朋友告诉他,她在电影院等他,小明过来的路线必须满足给定的规则: 1.假设小明在的位置是1号点,女朋友在的位置是n号点,则他们之间有n-2个点可以走,小明每次走的时候只能走到比当前所在点编号大的位置: 2.小明来的时候不能按一定的顺序经过某些地方.比如,如果女朋友告诉小明不能经过1 -> 2 -> 3,那么就要求小明来的时候走过的路径不能包含有1 -> 2 ->

hdu4511小明系列故事——女友的考验(ac自动机+最短路)

链接 预处理出来任意两点的距离,然后可以顺着trie树中的节点走,不能走到不合法的地方,另开一维表示走到了哪里,依次来更新. 注意判断一下起点是不是合法. 1 #include <iostream> 2 #include<cstdio> 3 #include<cstring> 4 #include<algorithm> 5 #include<stdlib.h> 6 #include<vector> 7 #include<cmat

HDU ACM 4511 小明系列故事——女友的考验-&gt;AC自动机+DP

分析:参考别人的搞. 1.AC自动机: 使用AC自动机来压缩路段,如禁掉的路段是1->2->3,那么插入字符串(123) ,注意点只有1~50,所以0~50用ASCII 压缩成字符串即可. 这样就能够完成禁止路段的在线状态转移. 2.DP部分: 两点之间的最短路.dp[i][j]表示在地点i,当前字符是j的状态. 初始化:fill(&dp0][0],&dp[maxn-1][maxp-1],inf),inf=1e12,memset不能用. 边界:首先找出出发点在Pool中的位置,

hdu4511---小明系列故事——女友的考验(AC自动机+dp)

小明系列故事--女友的考验 Time Limit: 500/200 MS (Java/Others) Memory Limit: 65535/32768 K (Java/Others) Total Submission(s): 863 Accepted Submission(s): 192 Problem Description 终于放寒假了,小明要和女朋友一起去看电影.这天,女朋友想给小明一个考验,在小明正准备出发的时候,女朋友告诉他,她在电影院等他,小明过来的路线必须满足给定的规则: 1.假

hdu 4511 小明系列故事——女友的考验

小明系列故事——女友的考验 Time Limit: 500/200 MS (Java/Others)    Memory Limit: 65535/32768 K (Java/Others)Total Submission(s): 822    Accepted Submission(s): 176 Problem Description 终于放寒假了,小明要和女朋友一起去看电影.这天,女朋友想给小明一个考验,在小明正准备出发的时候,女朋友告诉他,她在电影院等他,小明过来的路线必须满足给定的规则

HDU 4511 小明系列故事——女友的考验 (AC自动机+DP)

小明系列故事--女友的考验 Time Limit: 500/200 MS (Java/Others)    Memory Limit: 65535/32768 K (Java/Others)Total Submission(s): 1734    Accepted Submission(s): 466 Problem Description 终于放寒假了,小明要和女朋友一起去看电影.这天,女朋友想给小明一个考验,在小明正准备出发的时候,女朋友告诉他,她在电影院等他,小明过来的路线必须满足给定的规

AC自动机 + 二维最短路 HDU 4511 小明系列故事――女友的考验

这个题还是比较好想的. 首先将所有不可行方案建立AC自动机,然后跑最短路. 首先将小明放在(sta = 0,pos = 0)处,sta表示AC自动机上点的编号,pos表示坐标点的编号. 根据pos枚举下一次可以到达的地方[pos+1,n],然后sta在自动机上移动,如果某一步会使sta位于有标记的节点,那么这一步是不可行. #include <iostream> #include<time.h> #include<stdio.h> #include<string.

【HDU 4511】小明系列故事——女友的考验(AC自动机+DP)

Problem Description 终于放寒假了,小明要和女朋友一起去看电影.这天,女朋友想给小明一个考验,在小明正准备出发的时候,女朋友告诉他,她在电影院等他,小明过来的路线必须满足给定的规则: 1.假设小明在的位置是1号点,女朋友在的位置是n号点,则他们之间有n-2个点可以走,小明每次走的时候只能走到比当前所在点编号大的位置: 2.小明来的时候不能按一定的顺序经过某些地方.比如,如果女朋友告诉小明不能经过1 -> 2 -> 3,那么就要求小明来的时候走过的路径不能包含有1 ->

HDU 4511 小明系列故事——女友的考验 (AC自动机 + DP)题解

题意:从 1 走到 n,要求所走路径不能出现给定的路径,求最短路 思路:因为要求不能出现给定路径,那么我可以求助ac自动机完成判断. 我们可以在build的时候标记哪些路径不能出现,显然下面这种表示后缀不能出现,那么他也不能出现 if(node[node[u].fail].cnt && u) node[u].cnt = 1; //都不能取 然后再把图建完整.因为如果一个路径不在Trie中有两种情况,一种是他可能是某个不能走的串的前缀,那么我就重新指向这个不能走的串,比如Trie中只有AT,