POJ - 3308 Paratroopers(最大流)

时间:2023-03-09 02:34:43
POJ - 3308 Paratroopers(最大流)

1、这道题学了个单词,product 还有 乘积 的意思。。

题意就是在一个 m*n的矩阵中,放入L个敌军的伞兵,而我军要在伞兵落地的瞬间将其消灭。现在我军用一种激光枪组建一个防御系统,这种枪可以安装在一行(或者一列),并且安装在不同行(或者不同列)的费用是不一样的,枪的攻击范围是一行(或者一列)。安装所有枪的费用是它们每个费用的“乘积”,现在求组建这个系统需要的最小费用。

2、与前面做的二分图的一道题有点相似(POJ - 3041 Asteroids(最小点覆盖数))。但是现在这道题在不同行(或者不同列)的费用不同,所以还是有区别的。(ps:也就是说,之前那个把费用都设为1,也可以用这道题的方法来做?应该是的。)

建图:与上述的题类似,把行、列分别拆出来,分别作为二分图的X集、Y集中的点即可。然后在所建的二分图上加入超级源点(_S)、超级汇点(_T),再求最大流。

为什么求的最大流保证费用最小呢?因为对应二分图(加入源点、汇点后)中,总是选的_S->u 和v->_T 中的费用小的(即流量小的),并且覆盖了中间所有边(即不加源点、汇点前的二分图中的所有边)(好像是~,未证明)。

这里因为求的是乘积,所以用对数转化下:建图时,流量为 log(cost)即对各个cost求自然对数;再求出最大流maxflow,最后求 e^maxflow。(因为a*b*c=e^(loga+logb+logc);)

下面对题目样例作个解释:可忽略。

POJ - 3308 Paratroopers(最大流)

图中最大流的那个图里,画圈圈的是这条边满流(自己起的名,例:2/2),也就是有可能选的某行或某列(我是这么理解的),也就是说有可能不选它(例:1.5/1.5)。

下面解释样例为什么不选1.5/1.5而选2/2(即不选第1列而选第1行),因为选第1列的话,必须同时选第4列,才能代替选第1行和第4行,经比较,费用变大,所以不选。

对这种怎么选的情况,举个栗子,如图:

POJ - 3308 Paratroopers(最大流)

上面的为可替代的情况,

下面的为不可替代的情况。

3、

3、ISAP邻接表形式:直接用的这个速度比较好的模板

#include<iostream>
#include<stdio.h>
#include<string.h>
#include<math.h>
using namespace std; const int MAXN=;//点数的最大值
const int MAXM=;//边数的最大值(加的双向加,所以是2倍,别忘了加上超级源点和超级汇点)
const int INF=0x3f3f3f3f;
struct Edge{
int to,next;
double cap,flow;
}edge[MAXM];//注意是MAXM
int tol;
int head[MAXN];
int gap[MAXN],dep[MAXN],pre[MAXN],cur[MAXN];
void init(){
tol=;
memset(head,-,sizeof(head));
}
//加边,单向图三个参数,双向图四个参数
void addedge(int u,int v,double w,double rw=){
edge[tol].to=v;edge[tol].cap=w;edge[tol].next=head[u];
edge[tol].flow=;head[u]=tol++;
edge[tol].to=u;edge[tol].cap=rw;edge[tol].next=head[v];
edge[tol].flow=;head[v]=tol++;
}
//输入参数:起点、终点、点的总数
//点的编号没有影响,只要输入点的总数
double sap(int start,int end,int N){
memset(gap,,sizeof(gap));
memset(dep,,sizeof(dep));
memcpy(cur,head,sizeof(head));
int u=start;
pre[u]=-;
gap[]=N;
double ans=;
while(dep[start]<N){
if(u==end){
double Min=INF;
for(int i=pre[u];i!=-;i=pre[edge[i^].to])
if(Min>edge[i].cap-edge[i].flow)
Min=edge[i].cap-edge[i].flow;
for(int i=pre[u];i!=-;i=pre[edge[i^].to]){
edge[i].flow+=Min;
edge[i^].flow-=Min;
}
u=start;
ans+=Min;
continue;
}
bool flag=false;
int v;
for(int i=cur[u];i!=-;i=edge[i].next){
v=edge[i].to;
if(edge[i].cap-edge[i].flow&&dep[v]+==dep[u]){
flag=true;
cur[u]=pre[v]=i;
break;
}
}
if(flag){
u=v;
continue;
}
int Min=N;
for(int i=head[u];i!=-;i=edge[i].next)
if(edge[i].cap-edge[i].flow&&dep[edge[i].to]<Min){
Min=dep[edge[i].to];
cur[u]=i;
}
gap[dep[u]]--;
if(!gap[dep[u]])return ans;
dep[u]=Min+;
gap[dep[u]]++;
if(u!=start)u=edge[pre[u]^].to;
}
return ans;
} int main(){
int T;
int m,n,L;
double ri,ci;
int _S,_T;//超级源点,超级汇点
int r,c; scanf("%d",&T);
while(T--){
init();
scanf("%d%d%d",&m,&n,&L);
_S=m+n;
for(int i=;i<m;++i){
scanf("%lf",&ri);
addedge(_S,i,log(ri));
}
_T=m+n+;
for(int i=;i<n;++i){
scanf("%lf",&ci);
addedge(m+i,_T,log(ci));
} for(int i=;i<L;++i){
scanf("%d%d",&r,&c);
addedge(r-,m+c-,INF);
} printf("%.4f\n",exp(sap(_S,_T,m+n+)));
}
return ;
}

ps:当时因为 product 没读懂题,就搜了一下,搜到了这个博客:POJ 3308 Paratroopers