CTSC 1999 家园 【网络流24题】星际转移

直接把每一个点,每一天拆成一个点。

然后每个点到下一天连$inf$的边。

然后把飞船的路径用容量为飞船容量的边连接。

然后跑网络流判断是否满流。

#include <queue>
#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std;

#define F(i,j,k) for (int i=j;i<=k;++i)
#define D(i,j,k) for (int i=j;i>=k;--i)
#define inf 0x3f3f3f3f
#define maxn 500005

int n,m,k,p[25],flag=0,S=0,T=maxn-1,ans=0;
int sta[25][25],num[25];
int h[maxn],to[maxn],ne[maxn],fl[maxn],fr[maxn],en=0;
int hash[25][500],cnt=0,dis[maxn];
queue <int> q;

void add(int a,int b,int c)
{
	to[en]=b;fr[en]=a;ne[en]=h[a];fl[en]=c;h[a]=en++;
	to[en]=a;fr[en]=b;ne[en]=h[b];fl[en]=0;h[b]=en++;
}

bool tell()
{
	memset(dis,-1,sizeof dis);
	while (!q.empty()) q.pop();
	dis[S]=0;q.push(S);
	while (!q.empty())
	{
		int x=q.front();q.pop();
		for (int i=h[x];i>=0;i=ne[i])
		{
			if (fl[i]>0&&dis[to[i]]==-1)
			{
				dis[to[i]]=dis[x]+1;
				q.push(to[i]);
			}
		}
	}
	if (dis[T]==-1) return false;
	return true;
}

int zeng(int k,int now)
{
	if (k==T) return now;
	int r=0;
	for (int i=h[k];i>=0&&now>r;i=ne[i])
		if (dis[k]+1==dis[to[i]]&&fl[i]>0)
		{
			int t=zeng(to[i],min(now-r,fl[i]));
			fl[i]-=t;fl[i^1]+=t;r+=t;
		}
	if (!r) dis[k]=-1;
	return r;
}

int main()
{
	memset(h,-1,sizeof h);
	scanf("%d%d%d",&n,&m,&k);
	F(i,1,m)
	{
		scanf("%d",&p[i]);
		scanf("%d",&num[i]);
		F(j,0,num[i]-1)
		{
			scanf("%d",&sta[i][j]);
			if (sta[i][j]==-1) sta[i][j]=n+1;
		}
	}
	F(i,0,n+1)
		F(j,0,205)
			hash[i][j]=++cnt;
	add(S,hash[0][0],k);
	add(hash[n+1][0],T,inf);
	for (int z=0;z<=200;++z)
	{
		int tmp;
		F(i,0,n+1)
			add(hash[i][z],hash[i][z+1],inf);
		F(i,1,m)
			add(hash[sta[i][z%num[i]]][z],hash[sta[i][(z+1)%num[i]]][z+1],p[i]);
		add(hash[n+1][z+1],T,inf);
		while (tell()) while (tmp=zeng(S,inf)) ans+=tmp;
		if (ans==k)
		{
			flag=1;
			printf("%d\n",z+1);
			break; 
		}
	}
	if (!flag) printf("%d\n",0);
}

  

posted @ 2017-04-21 15:14  SfailSth  阅读(282)  评论(0编辑  收藏  举报