题目

​ 对一个二分图的边染色,满足有相同端点的边的颜色一定不同;

​ 设最优染色为\(C\) ,你的染色为\(X\),只需要满足$ X \le 2^ {\lceil log \ C \rceil }$

​ \(n_1,n_2 \le 10^6 \ , \ m \le 5 \times 10^5\)

题解

  • 这题由于没有很好地分析条件,只写了一个\(n m \sqrt m\) 的二分图匹配暴力

  • 首先答案应该是\(max \ deg_i\)

    证明:

    显然\(C \ge deg_m\)

    设二分图的两部为\(X\)和\(Y\),\(deg_m = max \ deg_i\)

    考虑选出\(X\)部的\(deg_i = deg_m\)的点,在\(Y\)部去找到一个完备匹配:

    利用霍尔定理,如果不满足,一定存在\(X\)选了\(x\)个\(deg_i = deg_m\) 的点,但是在\(Y\)和它们连通的点只有\(y\)个\((y \lt x)\),根据鸽巢原理,这\(y\)个点之中一定存在有\(deg_j \ge \lceil \frac{x \times deg_m}{y}\rceil \gt deg_m\) ,矛盾

    所以只考虑\(deg_m\)的点,\(X\)一定有一个完配\(M_x\),\(Y\)一定有一个完配\(M_y\)

    考虑$M_x \cup M_y $ 形成的若干个连通块,由于所有点的度数<=2,那么一个连通块只能是:

    1. $M_x \cap M_y $ 的一条边

    2. 环 (每个点都满足\(deg_i =deg_m\))

    3. 一条简单路径(只有末尾的某个端点不满足\(deg \neq deg_m\))

      后两者显然都可以调整到满足所有的\(deg_i=deg_m\)的点都被选入匹配

    所以一次匹配\(deg_m\)至少-1,重复这样的匹配,即\(C \le deg_m\) ,证毕

  • 题意启示我们去二分,可以补出两个虚点使得左右的度数都为偶数

  • 做欧拉回路对边染不同的色就可以每次使得\(deg_m\)折半

    #include<bits/stdc++.h>
    
    using namespace std;
    
    const int N=2000010;
    int n1,n2,n,m,hd[N],o,now,d[N],vis[N],col[N],C;
    struct edge{int u,v,w;}e[N]; char gc(){
    static char*p1,*p2,s[1000000];
    if(p1==p2)p2=(p1=s)+fread(s,1,1000000,stdin);
    return(p1==p2)?EOF:*p1++;
    }
    int rd(){
    int x=0;char c=gc();
    while(c<'0'||c>'9')c=gc();
    while(c>='0'&&c<='9')x=(x<<1)+(x<<3)+c-'0',c=gc();
    return x;
    } struct Edge{int v,nt,w;}E[N<<1];
    void adde(int u,int v,int w){
    vis[w]=0;
    E[o]=(Edge){v,hd[u],w};hd[u]=o++;
    E[o]=(Edge){u,hd[v],w};hd[v]=o++;
    d[u]++;d[v]++;
    } void dfs(int u){
    for(int&i=hd[u];i;i=E[i].nt){
    if(vis[E[i].w])continue;
    int tmp=E[i].w;
    vis[tmp]=1;
    dfs(E[i].v);
    now^=1;vis[tmp]=now;
    }
    } void solve(int l,int r){
    o=1;for(int i=l;i<=r;++i)adde(e[i].u,e[i].v,i);
    int fg=0,cnt=m;
    d[n+1]=d[n+2]=0;
    for(int i=l;i<=r;++i){
    int u=e[i].u,v=e[i].v;
    if(d[u]>1||d[v]>1)fg=1;
    if(d[u]&1)adde(u,n+1,++cnt);
    if(d[v]&1)adde(v,n+2,++cnt);
    d[u]=d[v]=0;
    }
    if(d[n+1]&1)adde(n+1,n+2,++cnt);
    d[n+1]=d[n+2]=0;
    if(!fg){
    ++C;hd[n+1]=hd[n+2]=0;
    for(int i=l;i<=r;++i){
    col[e[i].w]=C;
    hd[e[i].u]=hd[e[i].v]=0;
    }
    return ;
    }
    now=2;
    for(int i=l;i<=r;++i)if(!vis[i]){
    dfs(e[i].u);
    } static edge tmp[N];int p1=l,p2=0,mid;
    for(int i=l;i<=r;++i)if(vis[i]&1)e[p1++]=e[i];else tmp[++p2]=e[i];
    mid=p1;for(int i=1;i<=p2;++i)e[p1++]=tmp[i]; solve(l,mid-1);
    solve(mid,r);
    } int main(){
    freopen("color.in","r",stdin);
    freopen("color.out","w",stdout);
    n1=rd();n2=rd();m=rd();
    for(int i=1;i<=m;++i){
    int u=rd(),v=rd();
    e[i]=(edge){u,v+n1,i};
    }
    n=n1+n2;
    solve(1,m);
    printf("%d\n",C);
    for(int i=1;i<=m;++i)printf("%d\n",col[i]);
    return 0;
    }

【JZOJ6206】【20190610】二分图边染色的更多相关文章

  1. 二分图点染色 BestCoder 1st Anniversary($) 1004 Bipartite Graph

    题目传送门 /* 二分图点染色:这题就是将点分成两个集合就可以了,点染色用dfs做, 剩下的点放到点少的集合里去 官方解答:首先二分图可以分成两类点X和Y, 完全二分图的边数就是|X|*|Y|.我们的 ...

  2. hdu 5285 二分图黑白染色

    题意:给出 n 个人,以及 m 对互不认识的关系,剩余的人都互相认识,要将所有人分成两组,组内不能有互不认识的人,要求每组至少有一人,并且第一组人数尽量多,问两组人数或不可能时单独输出 BC 48 场 ...

  3. UVA - 10004 Bicoloring(判断二分图——交叉染色法 / 带权并查集)

    d.给定一个图,判断是不是二分图. s.可以交叉染色,就是二分图:否则,不是. 另外,此题中的图是强连通图,即任意两点可达,从而dfs方法从一个点出发就能遍历整个图了. 如果不能保证从一个点出发可以遍 ...

  4. 2019 ACM/ICPC Asia Regional shanxia D Miku and Generals (二分图黑白染色+01背包)

    Miku is matchless in the world!” As everyone knows, Nakano Miku is interested in Japanese generals, ...

  5. cf 557D 二分图黑白染色

    题意:给出一个 n 点 m 边的图,问最少加多少边使其能够存在奇环,加最少边的情况数有多少种 奇环和偶环其实就是二分图的性质:二分图不存在奇环,所以只要判断这张图是否是二分图就行了: 如果本身就不是二 ...

  6. cojs 二分图计数问题1-3 题解报告

    OwO 良心的FFT练手题,包含了所有的多项式基本运算呢 其中一部分解法参考了myy的uoj的blog 二分图计数 1: 实际是求所有图的二分图染色方案和 我们不妨枚举这个图中有多少个黑点 在n个点中 ...

  7. 3 Steps(二分图)

    C - 3 Steps Time limit : 2sec / Memory limit : 256MB Score : 500 points Problem Statement Rng has a ...

  8. 虚拟化构建二分图(BZOJ2080 题解+浅谈几道双栈排序思想的题)

    虚拟化构建二分图 ------BZOJ2080 题解+浅谈几道双栈排序思想的题 本题的题解在最下面↓↓↓ 不得不说,第一次接触类似于双栈排序的这种题,是在BZOJ的五月月赛上. [BZOJ4881][ ...

  9. [POJ2942]Knights of the Round Table(点双+二分图判定——染色法)

    建补图,是两个不仇恨的骑士连边,如果有环,则可以凑成一桌和谐的打麻将 不能直接缩点,因为直接缩点求的是连通分量,点双缩点只是把环缩起来 普通缩点                             ...

随机推荐

  1. Spring Security的RBAC数据模型嵌入

    1.简介 ​ 基于角色的权限访问控制(Role-Based Access Control)作为传统访问控制(自主访问,强制访问)的有前景的代替受到广泛的关注.在RBAC中,权限与角色相关联,用户通过成 ...

  2. Windows10 上安装 Anaconda 后命令提示符(cmd)下无法执行(python / pip)命令解决方案

    原文:https://blog.csdn.net/qq_38644840/article/details/85064408 安装Anaconda后一段时间内能够在命令提示符(cmd)界面运行pytho ...

  3. python检测远程udp端口是否打开的代码

    研发过程,把开发过程较好的代码收藏起来,如下的代码内容是关于python检测远程udp端口是否打开的代码,希望对各朋友有较大帮助. import socketimport threadingimpor ...

  4. Jenkins详细教程

    大纲 1.背景 在实际开发中,我们经常要一边开发一边测试,当然这里说的测试并不是程序员对自己代码的单元测试,而是同组程序员将代码提交后,由测试人员测试: 或者前后端分离后,经常会修改接口,然后重新部署 ...

  5. CentOS7安装VNC

    #安装 yum -y install tigervnc-server 将配置表复制到etc .service 修改配置文件 vim /etc/systemd/system/vncserver@\:.s ...

  6. LINUX下查看日志信息

    Linux下grep显示多行信息标准unix/linux下的grep通过以下参数控制上下文 grep -C 5 foo file 显示file文件中匹配foo字串那行以及上下5行 例如 grep -C ...

  7. UML——从类图到C++

    简易软件开发流程 实践中,use case and description.class diagram与sequence diagram三者搭配,几乎是UML项目的基本类型,所以在分工或外包的设计文档 ...

  8. cmdb资产管理2

    新增资产 现在api服务端已经能获取到我们要做的操作了.接下来应该是补充获取操作后对应的程序编写 我们要做的是把post请求发过来的数据保存到数据库.我们创建repository 名字的app,并设计 ...

  9. git拉取远程分支并切换到该分支

    整理了五种方法,我常用最后一种,这五种方法(除了第4中已经写了fetch的步骤)执行前都需要执行git fetch来同步远程仓库 (1)git checkout -b 本地分支名 origin/远程分 ...

  10. 使用Arduino连接HC-SR04超声波距离传感器的方法

    距离传感器是机器人项目最有用的传感器之一. HC-SR04是一种便宜的超声波距离传感器,可以帮助您的机器人在房间周围导航.通过一些努力和一个额外的组件,它也可以用作测量设备.在这篇文章中,您将学习到通 ...