【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集成easypoi导出word换行处理过程

《springboot集成easypoi导出word换行处理过程》SpringBoot集成Easypoi导出Word时,换行符n失效显示为空格,解决方法包括生成段落或替换模板中n为回车,同时需确... 目录项目场景问题描述解决方案第一种:生成段落的方式第二种:替换模板的情况,换行符替换成回车总结项目场景s

SpringBoot集成redisson实现延时队列教程

《SpringBoot集成redisson实现延时队列教程》文章介绍了使用Redisson实现延迟队列的完整步骤,包括依赖导入、Redis配置、工具类封装、业务枚举定义、执行器实现、Bean创建、消费... 目录1、先给项目导入Redisson依赖2、配置redis3、创建 RedissonConfig 配

SpringBoot中@Value注入静态变量方式

《SpringBoot中@Value注入静态变量方式》SpringBoot中静态变量无法直接用@Value注入,需通过setter方法,@Value(${})从属性文件获取值,@Value(#{})用... 目录项目场景解决方案注解说明1、@Value("${}")使用示例2、@Value("#{}"php

SpringBoot分段处理List集合多线程批量插入数据方式

《SpringBoot分段处理List集合多线程批量插入数据方式》文章介绍如何处理大数据量List批量插入数据库的优化方案:通过拆分List并分配独立线程处理,结合Spring线程池与异步方法提升效率... 目录项目场景解决方案1.实体类2.Mapper3.spring容器注入线程池bejsan对象4.创建

线上Java OOM问题定位与解决方案超详细解析

《线上JavaOOM问题定位与解决方案超详细解析》OOM是JVM抛出的错误,表示内存分配失败,:本文主要介绍线上JavaOOM问题定位与解决方案的相关资料,文中通过代码介绍的非常详细,需要的朋... 目录一、OOM问题核心认知1.1 OOM定义与技术定位1.2 OOM常见类型及技术特征二、OOM问题定位工具

基于 Cursor 开发 Spring Boot 项目详细攻略

《基于Cursor开发SpringBoot项目详细攻略》Cursor是集成GPT4、Claude3.5等LLM的VSCode类AI编程工具,支持SpringBoot项目开发全流程,涵盖环境配... 目录cursor是什么?基于 Cursor 开发 Spring Boot 项目完整指南1. 环境准备2. 创建

Spring Security简介、使用与最佳实践

《SpringSecurity简介、使用与最佳实践》SpringSecurity是一个能够为基于Spring的企业应用系统提供声明式的安全访问控制解决方案的安全框架,本文给大家介绍SpringSec... 目录一、如何理解 Spring Security?—— 核心思想二、如何在 Java 项目中使用?——

SpringBoot+RustFS 实现文件切片极速上传的实例代码

《SpringBoot+RustFS实现文件切片极速上传的实例代码》本文介绍利用SpringBoot和RustFS构建高性能文件切片上传系统,实现大文件秒传、断点续传和分片上传等功能,具有一定的参考... 目录一、为什么选择 RustFS + SpringBoot?二、环境准备与部署2.1 安装 RustF

springboot中使用okhttp3的小结

《springboot中使用okhttp3的小结》OkHttp3是一个JavaHTTP客户端,可以处理各种请求类型,比如GET、POST、PUT等,并且支持高效的HTTP连接池、请求和响应缓存、以及异... 在 Spring Boot 项目中使用 OkHttp3 进行 HTTP 请求是一个高效且流行的方式。

java.sql.SQLTransientConnectionException连接超时异常原因及解决方案

《java.sql.SQLTransientConnectionException连接超时异常原因及解决方案》:本文主要介绍java.sql.SQLTransientConnectionExcep... 目录一、引言二、异常信息分析三、可能的原因3.1 连接池配置不合理3.2 数据库负载过高3.3 连接泄漏