【JavaScript算法实践】1. 无向图连通分量问题

2024-04-14 01:48

本文主要是介绍【JavaScript算法实践】1. 无向图连通分量问题,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

【JavaScript算法实践】无向图连通分量问题

  • 1. 无向图连通分量
  • 2. 不符合条件的情况
  • 3. 连通分量计算
    • 方法一:深度优先搜索(DFS)
    • 方法二:广度优先搜索(BFS)
    • 方法三:并查集
    • 复杂度分析
  • 4. 经典题目
    • a. 岛屿问题
    • b. 门墙问题

1. 无向图连通分量

在开发马尔可夫分析模型功能的过程中,遇到了一个计算前检验模型是否合格的问题。其中最主要的就是要检验图中是否存在单独的节点,即多个连通分量。
什么是连通分量?
通俗地讲,在无向图中,若所有节点都是连通的(即任意选定两个节点,都存在至少一条路使得两个节点连通),则该图有且仅有一个连通子图,即它本身。此时称该图的连通分量为1。这便是马尔可夫过程模型需要检验的条件。

2. 不符合条件的情况

存在独立的节点
上图存在了单独的节点
连通分量大于1
上图连通分量大于1

3. 连通分量计算

方法一:深度优先搜索(DFS)

深度优先搜索的思路是最直观的。遍历图中所有节点,对于每个节点,如果该节点尚未被访问过,则从该节点开始深度优先搜索,通过邻接矩阵 isConnected 得到与该节点直接相连的节点有哪些,这些节点和该节点属于同一个连通分量,然后对这些节点继续深度优先搜索,直到同一个连通分量的所有节点都被访问到,即可得到一个节点。遍历全部节点以后,即可得到连通分量的总数。

/*** DFS深度优先遍历计算连通分量数* @param isConnected : 邻接矩阵 (二维数组)* @returns circles:连通分量数* nodeNum:节点数量 visited:已访问节点集  */
var findCircleNum = function(isConnected) {const nodeNum = isConnected.length;const visited = new Set();let circles = 0;for (let i = 0; i < nodeNum; i++) {if (!visited.has(i)) {dfs(isConnected, visited, nodeNum, i);circles++;}}return circles;
};const dfs = (isConnected, visited, nodeNum, i) => {for (let j = 0; j < nodeNum; j++) {if (isConnected[i][j] == 1 && !visited.has(j)) {visited.add(j);dfs(isConnected, visited, nodeNum, j);}}
};

方法二:广度优先搜索(BFS)

也可以通过广度优先搜索的方法得到连通分量的总数。对于每个节点,如果该节点尚未被访问过,则从该节点开始广度优先搜索,直到同一个连通分量中的所有节点都被访问到,即可得到一个连通分量。

/*** BFS广度优先遍历计算连通分量数* @param isConnected : 邻接矩阵 (二维数组)* @returns circles:连通分量数* nodeNum:节点数量 visited:已访问节点集 */ 
var findCircleNum = function(isConnected) {const nodeNum  = isConnected.length;const visited = new Set();let circles = 0;const queue = new Array();for (let i = 0; i < nodeNum ; i++) {if (!visited.has(i)) {queue.push(i);while (queue.length) {const j = queue.shift();visited.add(j);for (let k = 0; k < nodeNum; k++) {if (isConnected[j][k] === 1 && !visited.has(k)) {queue.push(k);}}}circles++;}}return circles;
};

方法三:并查集

计算连通分量数的另一个方法是使用并查集。初始时,每个节点都属于不同的连通分量。遍历矩阵 isConnected,如果两个节点之间有相连关系,则它们属于同一个连通分量,对它们进行合并。

遍历矩阵 isConnected 的全部元素之后,计算连通分量的总数。

/*** 并查集 计算连通分量数* @param isConnected * @returns circles:连通分量数* nodeNum:节点数量 visited:已访问节点集 */
var findCircleNum = function(isConnected) {const nodeNum = isConnected.length;const parent = new Array(nodeNum).fill(0).map((element, index) => index);for (let i = 0; i < nodeNum; i++) {for (let j = i + 1; j < nodeNum; j++) {if (isConnected[i][j] == 1) {union(parent, i, j);}}}let circles = 0;parent.forEach((element, index) => {if (element === index) {circles++;}});return circles;
};
// 并操作
const union = (parent, index1, index2) => {parent[find(parent, index1)] = find(parent, index2);
}
// 查操作
const find = (parent, index) => {if (parent[index] !== index) {parent[index] = find(parent, parent[index]);}return parent[index];
}

复杂度分析

方法时间复杂度空间复杂度
深度优先DFSO(n2)O(n)
广度优先BFSO(n2)O(n)
并查集O(n2logn)O(n)
其中,n是节点数量。对于DFS和BFS,时间上都需要遍历邻接矩阵内的所有元素,即n2,空间上都需要一个长度为n的数组来标记已经访问过的节点。
对于并查集,时间上除了遍历邻接矩阵内的所有元素外,若存在相连关系,最多可能有2n2次查找,和log(n2)次合并。故最坏的情况下复杂度为O(2n2log(n2)) = O(n2logn),空间上需要一个长度为n的数组来记录每个节点的祖先。

参考资料:

链接:https://leetcode-cn.com/problems/number-of-provinces/solution/sheng-fen-shu-liang-by-leetcode-solution-eyk0/
来源:力扣(LeetCode)

4. 经典题目

a. 岛屿问题

LeetCode.200 岛屿数量 【高频考题】
LeetCode.323 无向图中连通分量的数目
LeetCode.694 不同岛屿的数量
LeetCode.305 岛屿数量 II

b. 门墙问题

LeetCode.286 墙与门
LeetCode.542 01 矩阵
LeetCode.994 腐烂的橘子

这篇关于【JavaScript算法实践】1. 无向图连通分量问题的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Java学习手册之Filter和Listener使用方法

《Java学习手册之Filter和Listener使用方法》:本文主要介绍Java学习手册之Filter和Listener使用方法的相关资料,Filter是一种拦截器,可以在请求到达Servl... 目录一、Filter(过滤器)1. Filter 的工作原理2. Filter 的配置与使用二、Listen

Spring Boot中JSON数值溢出问题从报错到优雅解决办法

《SpringBoot中JSON数值溢出问题从报错到优雅解决办法》:本文主要介绍SpringBoot中JSON数值溢出问题从报错到优雅的解决办法,通过修改字段类型为Long、添加全局异常处理和... 目录一、问题背景:为什么我的接口突然报错了?二、为什么会发生这个错误?1. Java 数据类型的“容量”限制

Java对象转换的实现方式汇总

《Java对象转换的实现方式汇总》:本文主要介绍Java对象转换的多种实现方式,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录Java对象转换的多种实现方式1. 手动映射(Manual Mapping)2. Builder模式3. 工具类辅助映

SpringBoot请求参数接收控制指南分享

《SpringBoot请求参数接收控制指南分享》:本文主要介绍SpringBoot请求参数接收控制指南,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Spring Boot 请求参数接收控制指南1. 概述2. 有注解时参数接收方式对比3. 无注解时接收参数默认位置

SpringBoot基于配置实现短信服务策略的动态切换

《SpringBoot基于配置实现短信服务策略的动态切换》这篇文章主要为大家详细介绍了SpringBoot在接入多个短信服务商(如阿里云、腾讯云、华为云)后,如何根据配置或环境切换使用不同的服务商,需... 目录目标功能示例配置(application.yml)配置类绑定短信发送策略接口示例:阿里云 & 腾

关于MongoDB图片URL存储异常问题以及解决

《关于MongoDB图片URL存储异常问题以及解决》:本文主要介绍关于MongoDB图片URL存储异常问题以及解决方案,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐... 目录MongoDB图片URL存储异常问题项目场景问题描述原因分析解决方案预防措施js总结MongoDB图

SpringBoot项目中报错The field screenShot exceeds its maximum permitted size of 1048576 bytes.的问题及解决

《SpringBoot项目中报错ThefieldscreenShotexceedsitsmaximumpermittedsizeof1048576bytes.的问题及解决》这篇文章... 目录项目场景问题描述原因分析解决方案总结项目场景javascript提示:项目相关背景:项目场景:基于Spring

Spring Boot 整合 SSE的高级实践(Server-Sent Events)

《SpringBoot整合SSE的高级实践(Server-SentEvents)》SSE(Server-SentEvents)是一种基于HTTP协议的单向通信机制,允许服务器向浏览器持续发送实... 目录1、简述2、Spring Boot 中的SSE实现2.1 添加依赖2.2 实现后端接口2.3 配置超时时

解决Maven项目idea找不到本地仓库jar包问题以及使用mvn install:install-file

《解决Maven项目idea找不到本地仓库jar包问题以及使用mvninstall:install-file》:本文主要介绍解决Maven项目idea找不到本地仓库jar包问题以及使用mvnin... 目录Maven项目idea找不到本地仓库jar包以及使用mvn install:install-file基

Spring Boot读取配置文件的五种方式小结

《SpringBoot读取配置文件的五种方式小结》SpringBoot提供了灵活多样的方式来读取配置文件,这篇文章为大家介绍了5种常见的读取方式,文中的示例代码简洁易懂,大家可以根据自己的需要进... 目录1. 配置文件位置与加载顺序2. 读取配置文件的方式汇总方式一:使用 @Value 注解读取配置方式二