L-动物森友会

题目链接:https://ac.nowcoder.com/acm/contest/5278/L

解题思路:需要用到二分加网络流,二分我们需要的天数,然后就是建图

我们考虑到每天只能完成一定量的任务,因此我们可以在原点和星期几之间进行连线,边的容量是

二分出的天数包含的周次,和最后是否包含这一天,乘以这一星期的上限即可。然后就是在边和任务之间连线,

当然,我们可以完成无数次这个任务,因此链接为inf,最后是在任务和终点之间连线,这个是限制此任务最多完成多少次

#include<bits/stdc++.h>
using namespace std;
const int maxn=2e3+10;
const int inf=1e9+10;
struct no
{
    int next;
    int to;
    int val;
}edge[maxn];
int cnt;
int head[maxn];
int belong[maxn];
int dep[maxn];
int a[maxn];
int n,m,s,t;
int sum;
void add(int u,int v,int w)
{
    edge[cnt].to=v,edge[cnt].val=w,edge[cnt].next=head[u];
    head[u]=cnt++;
    edge[cnt].to=u,edge[cnt].val=0,edge[cnt].next=head[v];
    head[v]=cnt++;
}
bool bfs()
{
   memset(dep,0,sizeof(dep));
   dep[s]=1;
   queue<int>q;
   q.push(s);
   while(!q.empty())
   {
         int u=q.front();
         q.pop();
         for(int i=head[u];i!=-1;i=edge[i].next)
         {
              int v=edge[i].to;
           if(!dep[v]&&edge[i].val>0)
           {
             dep[v]=dep[u]+1;
             q.push(v);    
           }    
      }
   }
   return dep[t];
}
int dfs(int u,int maxflow)
{
    int tempflow;
    if(u==t)
    {
        return maxflow;
    }
    int add=0;
    for(int i=head[u];i!=-1;i=edge[i].next)
    {
        int v=edge[i].to;
        if(dep[v]==dep[u]+1&&edge[i].val>0&&(tempflow=dfs(v,min(maxflow-add,edge[i].val))))
        {
            edge[i].val-=tempflow;
            edge[i^1].val+=tempflow;
            add+=tempflow;
            if(maxflow==add) break;
        }
    }
    return add;
}
int dicnic()
{
    int ans=0;
    while(bfs())
    {
        int temp;
        while(temp=dfs(s,inf))  ans+=temp;
    }
    return ans;
}
vector<int>q[maxn]; 
int c[maxn];
int check(int mid)
{
    if(m*mid<sum) return 0;
    memset(head,-1,sizeof(head));
    cnt=0;
    int vc=0;
    s=++vc;
    t=++vc;
    for(int i=1;i<=7;i++) add(s,++vc,(mid/7+(mid%7>=i))*m);
    for(int i=1;i<=n;i++)
    {
        vc++;
        add(vc,t,c[i]);
        for(int j=0;j<q[i].size();j++)
        {
            add(q[i][j]+2,vc,inf);
        }
    }
    return dicnic()==sum;
}
int main()
{
cin>>n>>m;
for(int i=1;i<=n;i++)
{
    int qq;
    cin>>c[i]>>qq;
    sum+=c[i];
    for(int j=1;j<=qq;j++)
    {
        int ss;
        cin>>ss;
        q[i].push_back(ss);
    }
}
int l=1,r=1e9/m;
int res;
while(l<=r) {
        int mid=(l+r)>>1;
        if(check(mid)) r=mid-1,res=mid;
        else l=mid+1;
    }
    cout<<res<<endl;
return 0;
}

 

posted @ 2020-04-22 15:22  mcalex  阅读(192)  评论(0编辑  收藏  举报