c++全套流水账——染色法判断二分图,DFS的实践与应用

2023-11-05 17:30

本文主要是介绍c++全套流水账——染色法判断二分图,DFS的实践与应用,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

最近刷题比较多所以分享比较短也很少有视频了大家谅解一下。

染色法判断二分图

关于acwing

什么是二分图

二分图就是只你可以把一个图的点拉到左右两边。
这样它们就会变成两个集合。
那我们把原来图的边保留进这两个集合中。
最后如果我们可以使得这两个集合内部没有任何边,所有边都是两边互相连通的,我们称这个图为二分图。
否则不是二分图。
具体图片:
批注 2020-06-21 183219.png
这个图是二分图吗?
是的。
批注 2020-06-21 183313.png
批注 2020-06-21 183407.png
那这个呢?
不是。
批注 2020-06-21 183501.png
好了我们一会就证明为啥第二个一定不是。

二分图的特性

其实很简单,不能有奇数环。
啥是奇数环?
就是环上的点的数量是奇数。
很简单吧。
那我们是如何证明的呢?
请看下图。
我们把一个奇数环染色。
如果我们的这个是二分图:
那我们由一条边相连的两个点的数应该是不同的。
所以我们用01标完后我们就可以把标0的点放在1号集合。
标1的点放在2号集合。
这就很弱了对吧,我们给一个奇数环标一下就知道了。
批注 2020-06-21 185248.png

染色法判断二分图

依据上面这个原理,我们诞生了染色法。
我们可以把所有点染色撑出0,1。
那怎么染色呢?
没错就是dfs。
dfs搜索可以用来染色。
每次我们判断一下就好了。
判断一下如果我们的两个相邻的点染色相同,则这个图不是二分图。
那这个就很简单了。
我们来看一下具体模板题。
[模板]染色法判定二分图
本题题目:

给定一个n个点m条边的无向图,图中可能存在重边和自环。请你判断这个图是否是二分图。输入格式
第一行包含两个整数n和m。接下来m行,每行包含两个整数u和v,表示点u和点v之间存在一条边。输出格式
如果给定图是二分图,则输出“Yes”,否则输出“No”。数据范围
1≤n,m≤1e5
输入样例:
4 4
1 3
1 4
2 3
2 4
输出样例:
Yes

首先我们看一下数据范围。
数据范围是100000,我们再+一个10。

const int N = 100010;

或者:

const int N = 1e5 + 10;

接下来是读入。
这个就很简单了。
我们只要调用add函数存一个邻接表就行了。
注意是邻接表不是邻接矩阵!!!
一:

const int N = 100010;
int n, m, h[N], e[N], ne[N], idx, color[N];
void add(int a, int b)
{e[idx] = b, ne[idx] = h[a], h[a] = idx ++;
}

二:

cin >> n >> m;
memset(h, -1, sizeof h);
for(int i = 0; i < m; i ++)
{int a,b;cin >> a >> b;add(a, b);
}

memset那行很重要!如果一不小心忘写了……
就挖了……
批注 2020-06-21 204046.png
有一次就因为这玩意找了半个小时的bug,最后自闭了/误
然后我们来看看中间这个部分。
我们的的dfs有这么几个参数:

  1. u,表示需要染色的点
  2. c,表示染的颜色

所以:

bool dfs(int u, int c)

然后我们来想一下怎么判断。
我们遍历一遍所有点然后看一下染色结果就行了。
这个也没啥好讲的。

bool f = true;
for(int i = 1; i <= n; i ++)
{if(!color[i]){if(!dfs(i, 0)) {f = false;break;}}
}

接着我们来说一下dfs部分怎么写。
首先标记这个点。

color[u] = c;

接下来我们遍历一下。

for(int i = h[u]; i != -1; i = ne[i])

用j表示这个点。

int j = e[i];

接下来看一下这个点有没有染过色,如果没染过就继续往深里执行。
这不就是DFS吗?我好能水

if(!color[j])
{if(!dfs(j, 3 - c)) return false;
}

否则我们判断一下两个点染的色是否一样。
一样就返回false。

else if(color[j] == c) return false;

这里的color就是这个被遍历到的点染的色。
c就是这个点染的色。
所以判断这俩东西是否相等就行了。
最后如果这波都挺下来了就返回true。
dfs部分完整代码:

bool dfs(int u, int c)
{color[u] = c;for(int i = h[u]; i != -1; i = ne[i]){int j = e[i];if(!color[j]){if(!dfs(j, 3 - c)) return false;}else if(color[j] == c) return false;}return true;
}

输出我就不水了,大家看完整代码吧。

#include<bits/stdc++.h>using namespace std;const int N = 100010;int n, m;
int h[N], e[N], ne[N], idx;
int color[N];void add(int a, int b)
{e[idx] = b, ne[idx] = h[a], h[a] = idx ++;
}bool dfs(int u, int c)
{color[u] = c;for(int i = h[u]; i != -1; i = ne[i]){int j = e[i];if(!color[j]){if(!dfs(j, 3 - c)) return false;}else if(color[j] == c) return false;}return true;
}int main()
{cin >> n >> m;for(int i = 0; i < m; i ++){int a,b;cin >> a >> b;add(a, b);}bool f = true;for(int i = 1; i <= n; i ++){if(!color[i]){if(!dfs(i, 0)) {f = false;break;}}}if(f) cout << "Yes";else cout << "No";return 0;
}

好了这期分享就到这里了。

这里是我的全部分享

抱歉最近准备蓝桥杯青少组的比赛更新的比较慢(未来也会是这样)

大家谅解一下。

这期分享出的比较仓促也没有视频,非常抱歉。

这是作业,感谢@冷月无声re,差点忘了。

下期见!

(凑个整,250(wtcl))

这篇关于c++全套流水账——染色法判断二分图,DFS的实践与应用的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/351405

相关文章

C++中使用vector存储并遍历数据的基本步骤

《C++中使用vector存储并遍历数据的基本步骤》C++标准模板库(STL)提供了多种容器类型,包括顺序容器、关联容器、无序关联容器和容器适配器,每种容器都有其特定的用途和特性,:本文主要介绍C... 目录(1)容器及简要描述‌php顺序容器‌‌关联容器‌‌无序关联容器‌(基于哈希表):‌容器适配器‌:(

Python判断for循环最后一次的6种方法

《Python判断for循环最后一次的6种方法》在Python中,通常我们不会直接判断for循环是否正在执行最后一次迭代,因为Python的for循环是基于可迭代对象的,它不知道也不关心迭代的内部状态... 目录1.使用enuhttp://www.chinasem.cnmerate()和len()来判断for

JavaScript中的isTrusted属性及其应用场景详解

《JavaScript中的isTrusted属性及其应用场景详解》在现代Web开发中,JavaScript是构建交互式应用的核心语言,随着前端技术的不断发展,开发者需要处理越来越多的复杂场景,例如事件... 目录引言一、问题背景二、isTrusted 属性的来源与作用1. isTrusted 的定义2. 为

SpringBoot项目中Maven剔除无用Jar引用的最佳实践

《SpringBoot项目中Maven剔除无用Jar引用的最佳实践》在SpringBoot项目开发中,Maven是最常用的构建工具之一,通过Maven,我们可以轻松地管理项目所需的依赖,而,... 目录1、引言2、Maven 依赖管理的基础概念2.1 什么是 Maven 依赖2.2 Maven 的依赖传递机

Python调用另一个py文件并传递参数常见的方法及其应用场景

《Python调用另一个py文件并传递参数常见的方法及其应用场景》:本文主要介绍在Python中调用另一个py文件并传递参数的几种常见方法,包括使用import语句、exec函数、subproce... 目录前言1. 使用import语句1.1 基本用法1.2 导入特定函数1.3 处理文件路径2. 使用ex

Oracle查询优化之高效实现仅查询前10条记录的方法与实践

《Oracle查询优化之高效实现仅查询前10条记录的方法与实践》:本文主要介绍Oracle查询优化之高效实现仅查询前10条记录的相关资料,包括使用ROWNUM、ROW_NUMBER()函数、FET... 目录1. 使用 ROWNUM 查询2. 使用 ROW_NUMBER() 函数3. 使用 FETCH FI

在C#中获取端口号与系统信息的高效实践

《在C#中获取端口号与系统信息的高效实践》在现代软件开发中,尤其是系统管理、运维、监控和性能优化等场景中,了解计算机硬件和网络的状态至关重要,C#作为一种广泛应用的编程语言,提供了丰富的API来帮助开... 目录引言1. 获取端口号信息1.1 获取活动的 TCP 和 UDP 连接说明:应用场景:2. 获取硬

windos server2022里的DFS配置的实现

《windosserver2022里的DFS配置的实现》DFS是WindowsServer操作系统提供的一种功能,用于在多台服务器上集中管理共享文件夹和文件的分布式存储解决方案,本文就来介绍一下wi... 目录什么是DFS?优势:应用场景:DFS配置步骤什么是DFS?DFS指的是分布式文件系统(Distr

Java内存泄漏问题的排查、优化与最佳实践

《Java内存泄漏问题的排查、优化与最佳实践》在Java开发中,内存泄漏是一个常见且令人头疼的问题,内存泄漏指的是程序在运行过程中,已经不再使用的对象没有被及时释放,从而导致内存占用不断增加,最终... 目录引言1. 什么是内存泄漏?常见的内存泄漏情况2. 如何排查 Java 中的内存泄漏?2.1 使用 J

C++中实现调试日志输出

《C++中实现调试日志输出》在C++编程中,调试日志对于定位问题和优化代码至关重要,本文将介绍几种常用的调试日志输出方法,并教你如何在日志中添加时间戳,希望对大家有所帮助... 目录1. 使用 #ifdef _DEBUG 宏2. 加入时间戳:精确到毫秒3.Windows 和 MFC 中的调试日志方法MFC