leetcode547. 省份数量(dfs 并查集)

链接:https://leetcode-cn.com/problems/number-of-provinces/

题目

有 n 个城市,其中一些彼此相连,另一些没有相连。如果城市 a 与城市 b 直接相连,且城市 b 与城市 c 直接相连,那么城市 a 与城市 c 间接相连。

省份 是一组直接或间接相连的城市,组内不含其他没有相连的城市。

给你一个 n x n 的矩阵 isConnected ,其中 isConnected[i][j] = 1 表示第 i 个城市和第 j 个城市直接相连,而 isConnected[i][j] = 0 表示二者不直接相连。

返回矩阵中 省份 的数量。

用例

示例 1:

输入:isConnected = [[1,1,0],[1,1,0],[0,0,1]]
输出:2
示例 2:

输入:isConnected = [[1,0,0],[0,1,0],[0,0,1]]
输出:3

提示:

1 <= n <= 200
n == isConnected.length
n == isConnected[i].length
isConnected[i][j] 为 1 或 0
isConnected[i][i] == 1
isConnected[i][j] == isConnected[j][i]

思路

方法1
设置访问数组,dfs遍历每个未访问过的节点,计数

class Solution {
public:
	int findCircleNum(vector<vector<int>>& isConnected) {
		int n = isConnected.size(), ans = 0;
		vector<int>vis(n);
		for (int i = 0; i<n; ++i){
			if(!vis[i]){
				dfs(isConnected, vis, n, i);
				ans++;
			}
		}
		return ans;
	}
	void dfs(vector<vector<int>>&isConnected, vector<int>&vis, int n, int s){
		if (vis[s])
			return;
		vis[s] = 1;
		for (int i = 0; i<n; ++i){
			if (!vis[i] && isConnected[s][i] == 1){
				dfs(isConnected, vis, n, i);
			}
		}
	}
};

方法2
并查集
本题时标准的并查集模板题 ,对于需要求连通分量相关的题都可以使用并查集做
并查集是一种数据结构
并查集这三个字,一个字代表一个意思。
并(Union),代表合并
查(Find),代表查找
集(Set),代表这是一个以字典为基础的数据结构,它的基本功能是合并集合中的元素,查找集合中的元素
并查集的典型应用是有关连通分量的问题
并查集解决单个问题(添加,合并,查找)的时间复杂度都是O(1)O(1)
因此,并查集可以应用到在线算法中
并查集模板

class UnionFind{
public:
  //查找祖先
    int find(int x){
        int root = x;
        while(father[root] != -1){//当前root不是-1,即自己不是根节点时,继续找根节点
            root = father[root];
        }
        
        while(x != root){
            int original_father = father[x];//取普通父节点
            father[x] = root;//对于道路上每个祖先节点都改为root值 状态压缩
            x = original_father;
        }
        
        return root;
    }
    
    bool is_connected(int x,int y){//判断是否联通
        return find(x) == find(y);
    }
    
    void merge(int x,int y){//合并连通分量
        int root_x = find(x);
        int root_y = find(y);
        
        if(root_x != root_y){
            father[root_x] = root_y;
        }
    }
    
    void add(int x){//增加新的值
        if(!father.count(x)){
            father[x] = -1;
        }
    }
    
private:
    // 记录父节点
    unordered_map<int,int> father;
};

上一篇:邻接表的python实现与深度优先搜索


下一篇:LeetCode 547 省份数量(并查集学习)