bzoj 2744 [HEOI2012]朋友圈——补图!+匈牙利算法

题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2744

求最大的团<==>补图(有边的变成没边、没边的变成有边)的最大独立集!

A国的奇数和偶数变成两个团,B国变成一个二分图,A国和B国之间还有一些任意的边。

B国的部分肯定是求最大独立集。A国呢?A、B国的选点会互相影响。

其实枚举A国的选点情况就行了!每次把相关的B国点删掉,跑匈牙利。

观察数据范围,还专门分成两部分,一看就是一些复杂度在A国点上、一些复杂度在B国点上嘛!

思路。要敢于想补图。要能想到一些稍微暴力一点的方法,如枚举,而不是钻研如何权衡A国B国的选点情况什么的(==要回算时间复杂度)。

1A还是极好的。

时间复杂度玄学。不过匈牙利原来是边越多跑得越快呀。

复制代码
#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
const int N=205,M=3005;
int T,n,m,c,pos[N],cos[M],b[M],hd[N],xnt,p0,p1,c0,c1,ans,cnt,per[M];
bool bf[M][M],af[N][M],qx[M],vis[M];
struct Ed{
  int nxt,to;Ed(int n=0,int t=0):nxt(n),to(t) {}
}ed[N*M];
void add(int x,int y)
{
  ed[++xnt]=Ed(hd[x],y);hd[x]=xnt;
}
bool check(int i,int j)
{
  int k=(b[i]|b[j]),ct=0;
  while(k)k-=(k&-k),ct++;
  return ct&1;
}
bool dfs(int a)
{
  for(int i=c1;i<=m;i++)
    if(bf[a][i]&&!qx[i]&&!vis[i])
      {
    vis[i]=1;
    if(!per[i]||dfs(per[i]))
      {
        per[i]=a;return true;
      }
      }
  return false;
}
int xyl()
{
  int ret=0;memset(per,0,sizeof per);
  for(int i=1;i<=c0;i++)
    if(!qx[i]){
      memset(vis,0,sizeof vis);
      if(dfs(i))ret++;
    }
  return ret;
}
void solve()
{
  for(int i=1;i<=p0;i++)
    for(int j=p1;j<=n;j++)
      {
    memset(qx,0,sizeof qx);
    for(int k=hd[i];k;k=ed[k].nxt)qx[ed[k].to]=1;
    for(int k=hd[j];k;k=ed[k].nxt)qx[ed[k].to]=1;
    cnt=0;for(int i=1;i<=m;i++)if(!qx[i])cnt++;
    cnt-=xyl();ans=max(ans,cnt+2);
      }
  for(int i=1;i<=n;i++)
    {
      memset(qx,0,sizeof qx);
      for(int k=hd[i];k;k=ed[k].nxt)qx[ed[k].to]=1;
      cnt=0;for(int i=1;i<=m;i++)if(!qx[i])cnt++;
      cnt-=xyl();ans=max(ans,cnt+1);
    }
  memset(qx,0,sizeof qx);cnt=m;
  cnt-=xyl();ans=max(ans,cnt);
}
int main()
{
//  scanf("%d",&T);
//  while(T--)
//    {
      memset(hd,0,sizeof hd);xnt=0;ans=0;
      memset(bf,true,sizeof bf);memset(af,0,sizeof af);
      scanf("%d%d%d",&n,&m,&c);p0=0;p1=n+1;int x,y;
      for(int i=1;i<=n;i++)
    {
      scanf("%d",&x);
      if(x&1)pos[i]=++p0;
      else pos[i]=--p1;
    }
      c0=0;c1=m+1;
      for(int i=1;i<=m;i++)
    {
      scanf("%d",&b[i]);
      if(b[i]&1)cos[i]=++c0;
      else cos[i]=--c1;
      for(int j=1;j<i;j++)
        if(check(i,j))bf[cos[i]][cos[j]]=bf[cos[j]][cos[i]]=0;
    }
      while(c--)
    {
      scanf("%d%d",&x,&y);
      af[pos[x]][cos[y]]=1;
    }
      for(int i=1;i<=n;i++)
    for(int j=1;j<=m;j++)
      if(!af[i][j])add(i,j);
      solve();printf("%d\n",ans);
//    }
  return 0;
}
复制代码

 

posted on   Narh  阅读(396)  评论(0编辑  收藏  举报

编辑推荐:
· Linux系统下SQL Server数据库镜像配置全流程详解
· 现代计算机视觉入门之:什么是视频
· 你所不知道的 C/C++ 宏知识
· 聊一聊 操作系统蓝屏 c0000102 的故障分析
· SQL Server 内存占用高分析
阅读排行:
· DeepSeek V3 两周使用总结
· 回顾我的软件开发经历(1)
· C#使用yield关键字提升迭代性能与效率
· 低成本高可用方案!Linux系统下SQL Server数据库镜像配置全流程详解
· 4. 使用sql查询excel内容

导航

点击右上角即可分享
微信分享提示