LeetCode-827. 最大人工岛

最大人工岛

题目描述

原题链接

给你一个大小为 n x n 二进制矩阵 grid最多 只能将一格 0 变成 1

返回执行此操作后,grid 中最大的岛屿面积是多少?

岛屿 由一组上、下、左、右四个方向相连的 1 形成。

问题分析

为了方便实现并查集,将二维坐标转换为一维上的点,计算公式为idx = x * n + y

使用并查集维护所有grid[i][j] = 1的块连通性,并在维护连通性的过程种,使用sz[idx]记录下每个连通块的大小。

对原始的grid[i][j]进行处理:

  • grid[i][j] = 1,无需变为岛屿。直接计算连通块的最大面积sz[root],其中root(i, j)所属连通块的根节点编号。
  • grid[i][j] = 0,该位置可变为岛屿,统计四个方向连通位置是否存在岛屿,计算连通后的岛屿总和。

最后,对所有连通块大小取最大值。

程序代码

cpp 复制代码
class Solution {
private:
    int n;
    vector<int> p;  // 并查集
    vector<int> sz;  // 连通块的大小
    vector<int> dx = {-1, 0, 1, 0};
    vector<int> dy = {0, -1, 0, 1};

    // 将二维坐标转换为一维上的点
    int get(int x, int y) {
        return x * n + y;
    }

    // 并查集查找操作
    int find(int x) {
        if(p[x] != x)  p[x] = find(p[x]);
        return p[x];
    }

public:
    int largestIsland(vector<vector<int>>& grid) {
        n = grid.size();
        if(n == 0)  return 0;
        p = vector<int>(n * n, 0);
        sz = vector<int>(n * n, 1);  // 连通块大小初始化为1
        for(int i = 0; i < n * n; i++) {
            p[i] = i;
        }
        // 计算岛屿变化前的连通块大小
        for(int i = 0; i < n; i++) {
            for(int j = 0; j < n; j++) {
                if( grid[i][j] == 0)  continue;
                int a = get(i, j);
                // 遍历四个方向,计算连通块大小
                for(int t = 0; t < 4; t++) {
                    int x = i + dx[t], y = j + dy[t];
                    if(x < 0 || x >= n || y < 0 || y >= n || grid[x][y] == 0)  continue;
                    int b = get(x, y);
                    int pa = find(a), pb = find(b);
                    // 已经连通
                    if(pa == pb) {
                        continue;
                    }
                    // 集合合并
                    p[pb] = pa;
                    sz[pa] += sz[pb];
                }
            }
        }
        int res = 0;
        for(int i = 0; i < n; i++) {
            for(int j = 0; j < n; j++) {
                int a = get(i, j);
                if(grid[i][j] == 1) {
                    res = max(res, sz[find(a)]);
                }
                else {
                    int ans = 1;
                    unordered_set<int> s;
                    // 扩展四个方向的连通分量
                    for(int t = 0; t < 4; t++) {
                        int x = i + dx[t], y = j + dy[t];
                        if(x < 0 || x >= n || y < 0 || y >= n || grid[x][y] == 0)  continue;
                        int b = get(x, y);
                        int pb = find(b);
                        // 该连通分量已经扩展过了
                        if( s.count(pb) )  continue;
                        ans += sz[pb];
                        s.insert(pb);
                    }
                    res = max(res, ans);
                }
            }
        }
        return res;
    }
};

复杂度分析

时间复杂度为 O ( N 2 ) O(N^2) O(N2)

相关推荐
CHANG_THE_WORLD13 分钟前
switch语句在汇编层面的几种优化方式 ,为什么能进行优化
汇编·算法·switch·汇编分析·switch case·switch case 汇编·switch case 语句
山,离天三尺三14 分钟前
深度拷贝详解
开发语言·c++·算法
Blossom.11822 分钟前
把AI“撒”进农田:基于极值量化与状态机的1KB边缘灌溉决策树
人工智能·python·深度学习·算法·目标检测·决策树·机器学习
一只鱼^_40 分钟前
第 167 场双周赛 / 第 471 场周赛
数据结构·b树·算法·leetcode·深度优先·近邻算法·迭代加深
被制作时长两年半的个人练习生1 小时前
近期的笔试和面试的复盘
算法·面试·职场和发展·算子
测试老哥1 小时前
Postman环境变量设置全攻略
自动化测试·软件测试·python·测试工具·职场和发展·接口测试·postman
gsfl2 小时前
贪心算法1
算法·贪心算法
爱吃KFC的大肥羊2 小时前
第二次面试:C++qt开发实习生
面试·职场和发展
小猪咪piggy2 小时前
【算法】day8 二分查找+前缀和
算法
Word码2 小时前
[排序算法]希尔排序
c语言·数据结构·算法·排序算法