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);)

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

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

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

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

上面的为可替代的情况,

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

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

POJ - 3308 Paratroopers(最大流)的更多相关文章

  1. POJ 3308 Paratroopers 最大流,乘积化和 难度:2

    Paratroopers Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 7267   Accepted: 2194 Desc ...

  2. POJ 3308 Paratroopers(最小割EK(邻接表&矩阵))

    Description It is year 2500 A.D. and there is a terrible war between the forces of the Earth and the ...

  3. POJ 3308 Paratroopers(最小点权覆盖)(对数乘转加)

    http://poj.org/problem?id=3308 r*c的地图 每一个大炮可以消灭一行一列的敌人 安装消灭第i行的大炮花费是ri 安装消灭第j行的大炮花费是ci 已知敌人坐标,同时消灭所有 ...

  4. poj 3308 Paratroopers

    http://poj.org/problem?id=3308 #include <cstdio> #include <cstring> #include <algorit ...

  5. POJ - 3308 Paratroopers (最小点权覆盖)

    题意:N*M个格点,K个位置会有敌人.每行每列都有一门炮,能打掉这一行(列)上所有的敌人.每门炮都有其使用价值.总花费是所有使用炮的权值的乘积.求最小的总花费. 若每门炮的权值都是1,就是求最小点覆盖 ...

  6. POJ 3308 Paratroopers(最大流最小割の最小点权覆盖)

    Description It is year 2500 A.D. and there is a terrible war between the forces of the Earth and the ...

  7. POJ 3308 Paratroopers (对数转换+最小点权覆盖)

    题意 敌人侵略r*c的地图.为了消灭敌人,可以在某一行或者某一列安置超级大炮.每一个大炮可以瞬间消灭这一行(或者列)的敌人.安装消灭第i行的大炮消费是ri.安装消灭第j行的大炮消费是ci现在有n个敌人 ...

  8. poj 3308 Paratroopers(二分图最小点权覆盖)

    Paratroopers Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 8954   Accepted: 2702 Desc ...

  9. POJ 3308 Paratroopers(最小割EK)

    题目链接 题意 : 有一个n*m的矩阵,L个伞兵可能落在某些点上,这些点的坐标已知,需要在某些位置安上一些枪,然后每个枪可以将一行或者一列的伞兵击毙.把这种枪安装到不同行的行首.或者不同列的列首,费用 ...

随机推荐

  1. 大数据学习——flume日志分类采集汇总

    1. 案例场景 A.B两台日志服务机器实时生产日志主要类型为access.log.nginx.log.web.log 现在要求: 把A.B 机器中的access.log.nginx.log.web.l ...

  2. BZOJ 2561: 最小生成树【最小割/最大流】

    Description 给定一个边带正权的连通无向图G=(V,E),其中N=|V|,M=|E|,N个点从1到N依次编号,给定三个正整数u,v,和L (u≠v),假设现在加入一条边权为L的边(u,v), ...

  3. hdu 4091

    #include<stdio.h> #include<math.h> __int64 gcd(__int64 a,__int64 b) {  if(b==0)   return ...

  4. 可以学习相关框架【转:https://testerhome.com/topics/6283】

    https://testerhome.com/topics/6283 单元测试方面(Java): Junit:本来想用我熟悉的testng,但是开发的同学说测试springmvc只能用Junit.所以 ...

  5. 最小生成树求法 Prim + Kruskal

    prim算法的思路 和dijkstra是一样的 每次选取一个最近的点 然后去向新的节点扩张 注意这里的扩张 不再是 以前求最短路时候的到新的节点的最短距离 而是因为要生成一棵树 所以是要连一根最短的连 ...

  6. 前端的指导方针---css篇

    英语是渣渣,想学英语,又不想花钱报培训班.看不懂的文章,还是翻译一下留着自己看吧. 引自   :  https://github.com/bendc/frontend-guidelines HTML ...

  7. http_load分析(转)

    http://www.cnblogs.com/xuning/p/3954057.html 一.前言 http_load是一款测试web服务器性能的开源工具,从下面的网址可以下载到最新版本的http_l ...

  8. easyui combotree选项重复

    现象 编辑,赋值出现重复选项 原因 值之间有空格,比如我取值是3, 4, 6要改成3,4,6 注意:数值之间的空格去掉了

  9. 代理服务器squid简介

    Squid 是一个高性能.开源的代理缓存服务器和 Web 缓存进程,支持 FTP.Internet Gopher.HTTPS 和 SSL 等多种协议.它通过一个非阻塞的.I/O 事件驱动的单一进程处理 ...

  10. Tomcat+Servlet登录页面实例

    概念   Tomcat server是一个免费的开放源码的Web 应用server,属于轻量级应用server,在中小型系统和并发訪问用户不是非常多的场合下被普遍使用,是开发和调试JSP 程序的首选. ...