【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

相关文章

SpringBoot整合liteflow的详细过程

《SpringBoot整合liteflow的详细过程》:本文主要介绍SpringBoot整合liteflow的详细过程,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋...  liteflow 是什么? 能做什么?总之一句话:能帮你规范写代码逻辑 ,编排并解耦业务逻辑,代码

JavaSE正则表达式用法总结大全

《JavaSE正则表达式用法总结大全》正则表达式就是由一些特定的字符组成,代表的是一个规则,:本文主要介绍JavaSE正则表达式用法的相关资料,文中通过代码介绍的非常详细,需要的朋友可以参考下... 目录常用的正则表达式匹配符正则表China编程达式常用的类Pattern类Matcher类PatternSynta

Spring Security中用户名和密码的验证完整流程

《SpringSecurity中用户名和密码的验证完整流程》本文给大家介绍SpringSecurity中用户名和密码的验证完整流程,本文结合实例代码给大家介绍的非常详细,对大家的学习或工作具有一定... 首先创建了一个UsernamePasswordAuthenticationTChina编程oken对象,这是S

java实现docker镜像上传到harbor仓库的方式

《java实现docker镜像上传到harbor仓库的方式》:本文主要介绍java实现docker镜像上传到harbor仓库的方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录1. 前 言2. 编写工具类2.1 引入依赖包2.2 使用当前服务器的docker环境推送镜像2.2

Redis出现中文乱码的问题及解决

《Redis出现中文乱码的问题及解决》:本文主要介绍Redis出现中文乱码的问题及解决,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1. 问题的产生2China编程. 问题的解决redihttp://www.chinasem.cns数据进制问题的解决中文乱码问题解决总结

Java easyExcel实现导入多sheet的Excel

《JavaeasyExcel实现导入多sheet的Excel》这篇文章主要为大家详细介绍了如何使用JavaeasyExcel实现导入多sheet的Excel,文中的示例代码讲解详细,感兴趣的小伙伴可... 目录1.官网2.Excel样式3.代码1.官网easyExcel官网2.Excel样式3.代码

Java MQTT实战应用

《JavaMQTT实战应用》本文详解MQTT协议,涵盖其发布/订阅机制、低功耗高效特性、三种服务质量等级(QoS0/1/2),以及客户端、代理、主题的核心概念,最后提供Linux部署教程、Sprin... 目录一、MQTT协议二、MQTT优点三、三种服务质量等级四、客户端、代理、主题1. 客户端(Clien

Java中调用数据库存储过程的示例代码

《Java中调用数据库存储过程的示例代码》本文介绍Java通过JDBC调用数据库存储过程的方法,涵盖参数类型、执行步骤及数据库差异,需注意异常处理与资源管理,以优化性能并实现复杂业务逻辑,感兴趣的朋友... 目录一、存储过程概述二、Java调用存储过程的基本javascript步骤三、Java调用存储过程示

MySQL 中 ROW_NUMBER() 函数最佳实践

《MySQL中ROW_NUMBER()函数最佳实践》MySQL中ROW_NUMBER()函数,作为窗口函数为每行分配唯一连续序号,区别于RANK()和DENSE_RANK(),特别适合分页、去重... 目录mysql 中 ROW_NUMBER() 函数详解一、基础语法二、核心特点三、典型应用场景1. 数据分

Spring 框架之Springfox使用详解

《Spring框架之Springfox使用详解》Springfox是Spring框架的API文档工具,集成Swagger规范,自动生成文档并支持多语言/版本,模块化设计便于扩展,但存在版本兼容性、性... 目录核心功能工作原理模块化设计使用示例注意事项优缺点优点缺点总结适用场景建议总结Springfox 是