码迷,mamicode.com
首页 > 其他好文 > 详细

HDU 3491 最小点权割集

时间:2016-03-31 18:51:30      阅读:322      评论:0      收藏:0      [点我收藏+]

标签:

题意:有n个城市,m条双向边,有一群小偷从s前往t偷东西,警察叔叔们想要逮捕小偷们,现在告诉你在每座城市需要多少警察才能抓住这个城市的小偷,为什么说这个城市,因为小偷们会分开跑;然后题目还说不能在s和t逮捕小偷。问需要的最少警力是多少?

分析:这个问题可以变成这样:需要在哪些城市部署警力才能使得小偷不能从s到达t,也即最小点权割集。根据最小割=最大流(此处的最小割是指边权),我们可以这样建图。

建图:把除了s和t的每一个点拆开,在它们之间连一条单向边,权值为该点需要的警力。s和t同样拆开,不过权值为无穷大,因为题目说了不能在s和t逮捕。题目给出的边就按双向边相连,不过起点应该是这个点拆开后的终点,权值为无穷大。最后为了方便,可以把源点和s相连,t和汇点相连,权值都为无穷大。至此,就转换成最小割的模型,求出最大流即可。

手写isap写丑了,没有1A。

 

技术分享
 1 #include<stdio.h>
 2 #include<string.h>
 3 #define min(x,y) (x)<(y)?(x):(y)
 4 const int N=1111,M=1e5+1111,INF=0x3f3f3f3f;
 5 struct node
 6 {
 7     int v,c,next;
 8 }e[M];
 9 int gap[N],h[N],head[N];
10 int s,t,n,m,cnt;
11 void init()
12 {
13     memset(head,-1,sizeof(head));
14     cnt=0;
15 }
16 void add(int u,int v,int w)
17 {
18     e[cnt].v=v,e[cnt].c=w;
19     e[cnt].next=head[u],head[u]=cnt++;
20     e[cnt].v=u,e[cnt].c=0;
21     e[cnt].next=head[v],head[v]=cnt++;
22 }
23 int dfs(int u,int flow)
24 {
25     if(u==t)    return flow;
26     int a,i,v,c=flow,minh=t;
27     for(i=head[u];i!=-1;i=e[i].next)
28     {
29         if(e[i].c)
30         {
31             v=e[i].v;
32             if(h[v]==h[u]-1)
33             {
34                 a=min(c,e[i].c);
35                 a=dfs(v,a);
36                 e[i].c-=a;
37                 e[i^1].c+=a;
38                 c-=a;
39                 if(h[s]>t)    return flow-c;
40                 if(!c)    break;
41             }
42             minh=min(minh,h[v]);
43         }
44     }
45     if(c==flow)
46     {
47         if(--gap[h[u]]==0)    h[s]=t+1;
48         ++gap[h[u]=minh+1];
49     }
50     return flow-c;
51 }
52 int isap()
53 {
54     memset(gap,0,sizeof(gap));
55     memset(h,0,sizeof(h));
56     int ans=0;gap[0]=t+1;
57     while(h[s]<=t)
58         ans+=dfs(s,INF);
59     return ans;
60 }
61 int main()
62 {
63     int T,u,v,i,ss,tt;
64     scanf("%d",&T);
65     while(T--)
66     {
67         scanf("%d%d%d%d",&n,&m,&ss,&tt);
68         init();s=0,t=2*n+1;
69         for(i=1;i<=n;i++)
70         {
71             int x;
72             scanf("%d",&x);
73             if(i!=ss&&i!=tt)
74                 add(i,i+n,x);
75             else
76                 add(i,i+n,INF);
77         }
78         while(m--)
79         {
80             scanf("%d%d",&u,&v);
81             add(u+n,v,INF);
82             add(v+n,u,INF);
83         }
84         add(s,ss,INF);add(tt+n,t,INF);
85         printf("%d\n",isap());
86     }
87     return 0;
88 }
View Code

 

HDU 3491 最小点权割集

标签:

原文地址:http://www.cnblogs.com/L-King/p/5342077.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!