HDU 3873 Invade the Mars(带限制条件的Dijkstra)

题目网址:http://acm.hdu.edu.cn/showproblem.php?pid=3873

思路:

军队可以先等待在城市外面,等保护该城市的城市都被攻破后,直接进城(即进城不用耗费时间)。则进入该城市的最少时间为max(达到该城市的最少时间,到达保护该城市的所有城市的最大时间)。

用num[i]标记第i个城市被保护的数目,只有当该点被保护的数目为0时,才能入S集合,从而优化到其他点的时间。当前进入S集合的城市,遍历它所保护的城市,num[i]减一,记录下被保护的城市解除保护所需的最长时间。

说的有点绕,看代码会比较清楚。

需要注意的一点!!有重边,所以读边的时候要注意,如果是用邻接表就无所谓啦,但如果是用邻接矩阵 要读入最小值。

 

代码:

 1 #include <cstdio>
 2 #include <vector>
 3 #include <cstring>
 4 #include <algorithm>
 5 using namespace std;
 6 const int inf=1e8+5;
 7 int n,m;
 8 int num[3005];
 9 int dist[3005];
10 int S[3005];
11 int edge[3005][3005];
12 int rp[3005];//rp[i]表示攻破保护i城市的所有城市所需要的最长时间
13 vector<int>city[3005];
14 void init(){
15     memset(num, 0, sizeof(num));
16     memset(rp, 0, sizeof(rp));
17     memset(S, 0, sizeof(S));
18     for (int i=1; i<=n; i++) {
19         city[i].clear();
20         dist[i]=inf;
21     }
22     for (int i=1; i<=n; i++) {
23         for (int j=1; j<=n; j++) {
24             edge[i][j]=inf;
25         }
26     }
27 }
28 void dijkstra(){
29     dist[1]=0;
30     for(int i=0;i<n;i++){
31         int Min=inf,u=-1;
32         for (int j=1; j<=n; j++) {
33             dist[j]=max(dist[j], rp[j]);//每次遍历前  都要更新时间
34             if(dist[j]<Min && !num[j] && !S[j]){
35                 Min=dist[j];
36                 u=j;
37             }
38         }
39         if(u==-1)   return ;
40         S[u]=1;
41         for (int j=0; j<city[u].size(); j++) {//更新保护的城市信息
42             int t=city[u][j];
43             num[t]--;
44             rp[t]=max(rp[t],dist[u]);
45         }
46         city[u].clear();
47         for (int j=1; j<=n; j++) {
48             if(dist[j]>dist[u]+edge[u][j] && !S[j]){
49                 dist[j]=dist[u]+edge[u][j];
50             }
51         }
52     }
53 }
54 int main(){
55     int t;
56     scanf("%d",&t);
57     while (t--) {
58         scanf("%d%d",&n,&m);
59         init();
60         for(int i=0;i<m;i++){
61             int v,u,w;
62             scanf("%d%d%d",&v,&u,&w);
63             edge[v][u]=min(edge[v][u],w);//保留最小值
64             
65         }
66         for(int i=1;i<=n;i++){
67             int a,b;
68             scanf("%d",&a);
69             num[i]=a;//记录保护i城市的所有城市数量
70             for (int j=0; j<a; j++) {
71                 scanf("%d",&b);
72                 city[b].push_back(i);//city[b]为b城市 要保护的城市
73             }
74         }
75         dijkstra();
76         printf("%d\n",dist[n]);
77     }
78     return 0;
79 }

 

posted @ 2017-08-04 19:52  ventricle  阅读(286)  评论(0编辑  收藏  举报