LuoguP2762 太空飞行计划问题(最大权闭合子图,最小割)

时间:2023-01-17 01:57:25

题目描述

W 教授正在为国家航天中心计划一系列的太空飞行。每次太空飞行可进行一系列商业性实验而获取利润。现已确定了一个可供选择的实验集合E={E1,E2,…,Em},和进行这些实验需要使用的全部仪器的集合I={I1,I2,…In}。实验Ej需要用到的仪器是I的子集RjÍI。配置仪器Ik的费用为ck美元。实验Ej的赞助商已同意为该实验结果支付pj美元。W教授的任务是找出一个有效算法,确定在一次太空飞行中要进行哪些实验并因此而配置哪些仪器才能使太空飞行的净收益最大。这里净收益是指进行实验所获得的全部收入与配置仪器的全部费用的差额。

对于给定的实验和仪器配置情况,编程找出净收益最大的试验计划。

输入输出格式

输入格式:

第1行有2 个正整数m和n。m是实验数,n是仪器数。接下来的m 行,每行是一个实验的有关数据。第一个数赞助商同意支付该实验的费用;接着是该实验需要用到的若干仪器的编号。最后一行的n个数是配置每个仪器的费用。

输出格式:

第1 行是实验编号;第2行是仪器编号;最后一行是净收益。

解题思路:

相当于在实验和仪器都是点,都有自己的点权,实验为正,仪器为负。

实验向仪器有一条有向边。最后找到一个闭合子图使原图中不存在从这个子图中指向图外的边。

正点权与源点连权值,负点权与汇点相连,求正点权-最小割就是答案。

代码:

 #include<queue>
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
const int oo=0x3f3f3f3f;
struct pnt{
int hd;
int lyr;
int now;
}p[];
struct ent{
int twd;
int lst;
int vls;
}e[];
int cnt;
int n,m;
int s,t;
int sum;
char tmp[];
std::queue<int>Q;
void ade(int f,int t,int v)
{
cnt++;
e[cnt].twd=t;
e[cnt].vls=v;
e[cnt].lst=p[f].hd;
p[f].hd=cnt;
return ;
}
bool Bfs(void)
{
while(!Q.empty())
Q.pop();
for(int i=;i<=t;i++)
p[i].lyr=;
p[s].lyr=;
Q.push(s);
while(!Q.empty())
{
int x=Q.front();
Q.pop();
for(int i=p[x].hd;i;i=e[i].lst)
{
int to=e[i].twd;
if(p[to].lyr==&&e[i].vls>)
{
p[to].lyr=p[x].lyr+;
if(to==t)
return true;
Q.push(to);
}
}
}
return false;
}
int Dfs(int x,int fll)
{
if(x==t)
return fll;
for(int& i=p[x].now;i;i=e[i].lst)
{
int to=e[i].twd;
if(p[to].lyr==p[x].lyr+&&e[i].vls>)
{
int ans=Dfs(to,std::min(fll,e[i].vls));
if(ans>)
{
e[i].vls-=ans;
e[((i-)^)+].vls+=ans;
return ans;
}
}
}
return ;
}
int Dinic(void)
{
int ans=;
while(Bfs())
{
int dlt;
for(int i=;i<=t;i++)
p[i].now=p[i].hd;
while(dlt=Dfs(s,oo))
ans+=dlt;
}
return ans;
}
int main()
{
// freopen("a.in","r",stdin);
scanf("%d%d",&m,&n);
s=n+m+;
t=s+;
for(int i=;i<=m;i++)
{
int v;
scanf("%d",&v);
sum+=v;
ade(s,i,v);
ade(i,s,);
int len=;
memset(tmp,,sizeof(tmp));
std::cin.getline(tmp,);
while(sscanf(tmp+len,"%d",&v)==)
{
ade(i,v+m,oo);
ade(v+m,i,);
if(v==)len++;
else{
while(v)
{
len++;
v/=;
}
}
len++;
}
}
for(int i=;i<=n;i++)
{
int v;
scanf("%d",&v);
ade(i+m,t,v);
ade(t,i+m,);
}
int ans=Dinic();
for(int i=;i<=m;i++)
{
if(p[i].lyr>)
printf("%d ",i);
}
puts("");
for(int i=;i<=n;i++)
{
if(p[i+m].lyr>)
printf("%d ",i);
}
puts("");
printf("%d\n",sum-ans);
return ;
}