深圳幻海软件技术有限公司 欢迎您!

二分图(Bipartite Graph)

2023-05-30

一、简介二分图の定义        二分图又叫二部图,是图论中的一种特殊模型。    假设S=(V,E)是一个无向图。如果顶点V可分割为两个互不相交的子集(A,B),并且图中的每条

一、简介

二分图の定义

        二分图又叫二部图,是图论中的一种特殊模型。

        假设S=(V,E)是一个无向图。如果顶点V可分割为两个互不相交的子集(A,B),并且图中的每条边(i,j)所关联的两个顶点i和j分别属于这两个不同的顶点集(i in A,j in B),就可以称图S为一个二分图。简单来说,就是顶点集V可分割为两个互不相交的子集,并且图中每条边依附的两个顶点都分属于这两个互不相交的子集,两个子集内的顶点不相邻。

二分图の匹配

        给定一个二分图S,在S的一个子图M中,M的边集{E}中的任意两条边都不依附于同一个顶点,则称M是一个匹配。

        极大匹配是指在当前已完成的匹配下,无法再通过增加未完成匹配的边的方式来增加匹配的边数。

        最大匹配是所有极大匹配当中边数最大的一个匹配。

        选择这样的边数最大的子集称为图的最大匹配问题。如果一个匹配中,图中的每个顶点都和图中某条边相关联,则称此匹配为完全匹配,也称作完备匹配。
        完全匹配一定是极大匹配,但是极大匹配不一定是完全匹配。下图就是一个最大匹配。

 二分图の判断

       对于二分图的判断方法最常见的是染色法,就是对每一个点进行染色操作,只用黑白两种颜色,能不能使所有的点都染上了色且相邻两个点的颜色不同?如果可以那么这个图就是一个二分图,对于判断是否是一个二分图的方法可以用dfs和bfs两种方式去实现。代码参见“代码实现”部分。

二分图の最大匹配

        给定一个二分图S,在S的一个子图M中,M的边集中的任意两条边都不依附于同一个顶点,则称M是一个匹配。选择这样的边数最大的子集称为图的最大匹配问题。

        首先要了解两个概念:

|交替路|从一个未匹配的点出发,依次经过未匹配边、匹配边、未匹配边....

|增广路|从一个未匹配的点出发,走交替路,到达了一个未匹配过的点。  

        对于求二分图最大匹配的算法可以用网络流去跑一个最大流求解,还可以用二分图的常见算法匈牙利算法,这个算法的核心就是去找未匹配的点,然后从这个点出发去寻找增广路,因为增广路有几个主要的特点:

  • 增广路有奇数条边 。
  • 路径上的点一定是一个在X边,一个在Y边,交错出现。
  • 起点和终点都是目前还没有配对的点。
  • 未匹配边的数量比匹配边的数量多1。

        重点是第4点,我们可以根据此特性,把这条增广路中的匹配边改为未匹配边,把未匹配边改为匹配边,这样我们就可以使总匹配边数+1了,根据上面的图得出下图,很显然匹配边多了一条。

        代码见“代码实现”部分。

二、代码实现

“二分图の判断”配套代码:

BFS判断二分图

  1. vector<int>G[maxn];//存边
  2. int col[maxn];//标记顶点颜色
  3. int n,m;//点和边的个数
  4. bool bfs()
  5. {
  6. queue<int>q;
  7. q.push(1);//放入第一个点
  8. memset(col,0,sizeof(col));
  9. col[1]=1;//先标记第一个点为1
  10. while(!q.empty())
  11. {
  12. int v=q.front();
  13. q.pop();
  14. for(int i=0; i<G[v].size(); i++)
  15. {
  16. int xx=G[v][i];
  17. if(col[xx]==0)//判断这个点是否标记过
  18. {
  19. col[xx]=-col[v];//没有标记过就标记上与v相反的颜色
  20. q.push(xx);
  21. }
  22. else
  23. {
  24. if(col[v]==col[xx])//如果颜色冲突说明不是二分图
  25. {
  26. return false;
  27. }
  28. }
  29. }
  30. }
  31. return true;
  32. }

“二分图の最大匹配”配套代码

匈牙利算法代码

  1. vector<int> G[maxn];//存边
  2. int pre[maxn];//记录匹配点
  3. bool vis[maxn];//标记是否匹配过
  4. int n,m;//n个点 m条边
  5. bool dfs(int x)
  6. {
  7. for(int i=0; i<G[x].size(); i++)
  8. {
  9. int v=G[x][i];
  10. if(vis[v]==false)//判断是否标记过
  11. {
  12. vis[v]=true;
  13. if(pre[v]==-1||dfs(pre[v]))// 判断当前点是否匹配过 dfs为判断这个点能不能和其他的点匹配
  14. {
  15. pre[v]=x;
  16. return true;
  17. }
  18. }
  19. }
  20. return false;
  21. }
  22. int Fun()
  23. {
  24. int sum=0;
  25. memset(pre,-1,sizeof(pre));
  26. for(int i=1; i<=n; i++)
  27. {
  28. memset(vis,false,sizeof(vis));//每次遍历都需要初始化
  29. if(dfs(i))
  30. {
  31. sum++;
  32. }
  33. }
  34. return sum;
  35. }


参考文献:

干货|二分图详解https://www.zhihu.com/tardis/landing/m/360/art/89972891以上内容就是本文的全部内容啦!感谢阅读!

文章知识点与官方知识档案匹配,可进一步学习相关知识
算法技能树首页概览47303 人正在系统学习中