超级大水题

Description

 史努比最爱水题了,他知道现在他要挑战的是一道超级大水题哦。不过因为太水了,而且他又很懒,所以把这道水题推给了你们。

 

有n个(不包括ZSTU和火车东站)车站介于ZSTU和火车东站之间,一共有m辆公交车往返于这些车站和火车东站。每个车站假设可以站无数个人,每辆车最多承载H[i]个人,经过S[i]个车站。假设一辆车依次经过1,2,3,这三个车站,那么它将一直往复1->2->3->1->2...现在认为车从一个站到下一个站要花费1个小时,人只能在车站或ZSTU,火车东站上下车,初始时有k个人在ZSTU(0站),所有的车都在初始站(所给车站的第一个车站为初始站),现在问你们最快让全部人从ZSTU到火车东站要多少小时。所有车同时运行。

 

Input

 第一行有一个T<=10,表示T组测试样例,第二行三个数n(车站数目),m(车数目),k(人数),(n<=13,m<=20,k<=50);接下来的m行给出车信息,每行给出H[i](能承载的最大人数),S[i](经过的车站数目),还有依次给出S[i]个车站编号。ZSTU用0编号表示,火车东站用-1表示。

 

Output

 若无解则输出-1,否则逐行输出答案。

 

Sample Input

2 2 2 1 1 3 0 1 2 1 3 1 2 -1 2 3 3 1 2 0 2 1 2 1 2 1 2 1 -1

Sample Output

5 7

HINT

 

 第一个样例,一人从0站乘坐1号公交车途径1站,到2站下车,花费2S;此时2号车位于-1站,再经过2S,2号车从-1站途径1站到2站,人上2号车再经过1S到达-1站,共花费5S。或者一人从0站乘坐一号公交车到1站下车,花费1S,此时2号车位于2站,再经过2S到1站,人上2号车再经过2S到达-1站,共花费5S。

 

题解:

首先判断学校到火车东站能不能到站,即是否在同一个集团里面,用并查集就可以了。然后跑网络流,网络流跑一次后会更改原来储存的边的信息,因此可以枚举时间进行网络流,按照时间建边,在上一个时间段跑网络流的基础上,继续建图在继续跑网络流,将每次的最大流加起来直到总和大于等于k就可以返回当前天数了。

具体操作是将第i-1小时的第j个车的信息向第i小时的第j个车进行连边,容量为INF,将每辆车第i-1小时所待的公交车站向第i小时所待的公交车站连边,容量为车的最大载人数。

#include <bits/stdc++.h>
//CLOCKS_PER_SEC
#define se second
#define fi first
#define ll long long
#define lson l,m,rt<<1
#define rson m+1,r,rt<<1|1
#define Pii pair<int,int>
#define Pli pair<ll,int>
#define ull unsigned long long
#define pb push_back
#define fio ios::sync_with_stdio(false);cin.tie(0)
const double Pi=3.14159265;
const int N=1e3+5;
const ull base=163;
const int INF=0x3f3f3f3f;
using namespace std;
int n,m,k,s,t;
int head[10100],to[100010],cur[10100],nx[100010];
int cap[100010];
int tot=0;
void add(int x,int y,int c){
    to[tot]=y;
    nx[tot]=head[x];
    cap[tot]=c;
    head[x]=tot++;
    
    to[tot]=x;
    nx[tot]=head[y];
    cap[tot]=0;
    head[y]=tot++;
}
void init(int n){
    tot=0;
    memset(head,-1,sizeof(head));
}
int d[N];
int bfs(){
    memset(d,-1,sizeof(d));
    queue<int>q;
    q.push(s);
    d[s]=1;
    while(!q.empty()){
        int u=q.front();q.pop();
        for(int i=head[u];~i;i=nx[i]){
            int v=to[i];
            if(d[v]==-1&&cap[i]>0){
                d[v]=d[u]+1;
                q.push(v);
            }
        }
    }
    return d[t]!=-1;
}
int dfs(int s,int a){
    if(s==t||a==0)return a;
    int flow=0,f;
    for(int &i=cur[s];~i;i=nx[i]){
        int v=to[i];
        if(d[s]+1==d[v] && cap[i]>0 && (f=dfs(v,min(a,cap[i])))>0){
            flow+=f;
            cap[i]-=f;
            cap[i^1]+=f;
            a-=f;
            if(a==0)break;
        }
    }
    return flow;
}
int dinic(){
    int ans=0;
    while(bfs()){
        for(int i=0;i<=1000;i++)cur[i]=head[i];
        while(int di=dfs(s,INF)){
            ans+=di;
        }
    }
    return ans;
}
int ship[N][N];
int num[N];
int r[N];
int fa[N];
int F(int x){
    return fa[x]==x?x:fa[x]=F(fa[x]);
}

bool check(int x,int y){
    if(F(x)==F(y))return 1;
    else return 0;
}
int cal(int a,int d){
    return d*n+a;
}
void solve(){
    int flow=0;
    int i;
    init(n);// n -1     n-1   0
    add(s,cal(n-1,0),INF);
    add(cal(n,0),t,INF);
    for(i=1;flow<k;i++){
        add(s,cal(n-1,i),INF);add(cal(n,i),t,INF);
        for(int j=1;j<=n;j++){
            add(cal(j,i-1),cal(j,i),INF);
        }
        for(int j=1;j<=m;j++){
            int a=(i-1)%num[j]+1;
            int b=i%num[j]+1;
            add(cal(ship[j][a],i-1),cal(ship[j][b],i),r[j]);
        }
        flow+=dinic();
        
    }
   printf("%d\n",i-1);
}
int main(){
	 // clock_t start, end;
    int T;scanf("%d",&T);
    while(T--){
        s=0,t=1000;
        scanf("%d%d%d",&n,&m,&k);
        for(int i=0;i<=n+2;i++)fa[i]=i;
        for(int i=1;i<=m;i++){
            scanf("%d%d",&r[i],&num[i]);
            for(int j=1;j<=num[i];j++){
                scanf("%d",&ship[i][j]);
                if(ship[i][j]==0)ship[i][j]=n+1;
                if(ship[i][j]==-1)ship[i][j]=n+2;
                if(j>1){
                    fa[F(ship[i][j-1])]=F(ship[i][j]);
                }
            }
        }
        n+=2;
        if(check(n, n-1)){
            solve();
        }
        else{
            printf("0\n");
        }
    }
    // end = clock();
    // cout << "time :"<<(double)(end - start) / CLOCKS_PER_SEC << endl;
    return 0;
}
/*

*/

 

全部评论

相关推荐

牛客101244697号:电竞协会感觉不如不写
点赞 评论 收藏
分享
威猛的小饼干正在背八股:挂到根本不想整理
点赞 评论 收藏
分享
12-06 20:47
已编辑
复旦大学 C++
华为 终端小艺 定级估计是15a
khj:只要家里条件还行和不愿意太卷真别去华为这种农村做题家云集的地方
点赞 评论 收藏
分享
评论
点赞
收藏
分享
牛客网
牛客企业服务