luogu P3153 [CQOI2009]跳舞

Posted shxnb666

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了luogu P3153 [CQOI2009]跳舞相关的知识,希望对你有一定的参考价值。

题面

https://www.luogu.org/problemnew/show/P3153

题解:

水题。

二分答案+最大流检验。

对于每个人拆点,分人和不喜欢的人。

我认为$yyb$一开始说的分人、喜欢的人、不喜欢的人与正解等价,因为我一开始想的就是这个。

但是通过合并点优化到了正解。

  • 注意:网络流中合并点的小技巧引用。
#include<cstdio>
#include<iostream>
#include<cstring>
#include<vector>
#include<queue>
#define ri register int
#define N 505
#define INF 1000000007
#define T (4*n+1)

using namespace std;

int n,k,x;
vector<int> to,w;
vector<int> ed[N];
int cur[N],d[N];
char g[N][N];

void add_edge(int u,int v,int w1,int w2) 
  to.push_back(v); w.push_back(w1); ed[u].push_back(to.size()-1);
  to.push_back(u); w.push_back(w2); ed[v].push_back(to.size()-1);


bool bfs() 
  queue<int> q;
  memset(d,0x3f,sizeof(d));
  d[0]=0; q.push(0);
  while (!q.empty()) 
    int x=q.front(); q.pop();
    for (ri i=0,l=ed[x].size();i<l;i++) 
      int e=ed[x][i];
      if (w[e] && d[x]+1<d[to[e]]) 
        d[to[e]]=d[x]+1;
        q.push(to[e]);
      
    
  
  return d[T]<=1000000;


int dfs(int x,int limit) 
  if (x==T || !limit) return limit;
  int tot=0;
  for (ri &i=cur[x];i<ed[x].size();i++) 
    int e=ed[x][i];
    if (d[to[e]]==d[x]+1 && w[e]) 
      int f=dfs(to[e],min(limit,w[e]));
      if (!f) continue;
      w[e]-=f; w[1^e]+=f; 
      tot+=f; limit-=f;
      if (!limit) return tot;
    
  
  return tot;


int dinic() 
  int ret=0;
  while (bfs()) 
    memset(cur,0,sizeof(cur));
    ret+=dfs(0,INF);
  
  return ret;


inline int bl(int x) return x;
inline int bd(int x) return n+x;
inline int gd(int x) return 2*n+x;
inline int gl(int x) return 3*n+x;

bool can(int x) 
  to.clear(); w.clear();
  for (ri i=0;i<=T;i++) ed[i].clear();
  for (ri i=1;i<=n;i++) add_edge(0,bl(i),x,0);
  for (ri i=1;i<=n;i++) add_edge(bl(i),bd(i),k,0);
  for (ri i=1;i<=n;i++) add_edge(gd(i),gl(i),k,0);
  for (ri i=1;i<=n;i++) add_edge(gl(i),T,x,0);
  for (ri i=1;i<=n;i++) 
    for (ri j=1;j<=n;j++) if (g[i][j]==Y) add_edge(bl(i),gl(j),1,0); else add_edge(bd(i),gd(j),1,0);
  if (dinic()==x*n) return 1; else return 0;


int main() 
  scanf("%d %d",&n,&k);
  for (ri i=1;i<=n;i++) scanf("%s",g[i]+1);
  int lb=0,rb=n;
  int ans=-1;
  while (lb<=rb) 
    int mid=(lb+rb)/2;
    if (can(mid)) ans=mid,lb=mid+1; else rb=mid-1;
  
  cout<<ans<<endl;

 

以上是关于luogu P3153 [CQOI2009]跳舞的主要内容,如果未能解决你的问题,请参考以下文章

luogu P3153 [CQOI2009]跳舞 |网络流最大流

AC日记——[CQOI2009]DANCE跳舞 洛谷 P3153

P3153 [CQOI2009]跳舞

P3153 [CQOI2009]跳舞 二分网络流

[CQOI2009]DANCE跳舞(ISAP写法)

bzoj1305[CQOI2009]dance跳舞