在二维地图上, 0代表海洋, 1代表陆地,我们最多只能将一格 0 海洋变成 1变成陆地。

进行填海之后,地图上最大的岛屿面积是多少?(上、下、左、右四个方向相连的 1 可形成岛屿)

示例 1:

输入: [[1, 0], [0, 1]]
输出: 3
解释: 将一格0变成1,最终连通两个小岛得到面积为 3 的岛屿。

示例 2:

输入: [[1, 1], [1, 0]]
输出: 4
解释: 将一格0变成1,岛屿的面积扩大为 4。

示例 3:

输入: [[1, 1], [1, 1]]
输出: 4
解释: 没有0可以让我们变成1,面积依然为 4。

说明:

  • 1 <= grid.length = grid[0].length <= 50
  • 0 <= grid[i][j] <= 1

思路:

通过题目我们寻找一种遍历的方法,对每个数字0,判断如果将其填海,面积为多少,记为area

用中文叙述就是:

area初始化为0
如果此海上面为陆地,area+上面连成片的陆地面积。
如果此海右面为陆地,area+右面连成片的陆地面积。
如果此海下面为陆地,area+下面连成片的陆地面积。
如果此海左面为陆地,area+左面连成片的陆地面积。

代码表示则是:

if(g[i-1][j]==1)area+=上面连成片的陆地面积;
if(g[i][j+1]==1)area+=右面连成片的陆地面积;
if(g[i+1][j]==1)area+=下面连成片的陆地面积;
if(g[i][j-1]==1)area+=左面连成片的陆地面积;

本题的目的就是找到这个最大的area

可是遍历的时候会有两个麻烦点

  1. 如果上下左右为连成一片的陆地呢,那按照上面的公式则会重复计算面积
  2. 边界判断

其实2好解决,每次查询的时候进行边界检查即可。

但是1的解决方式就很复杂,我的解决方式是通过增加两个记录数组,给连成片的陆地进行编号,计算上下左右的陆地面积,若遇到有相同的陆地编号则不进行计算。

unsigned tag[50][50]={0};
/*用来记录第X,Y块属于哪块陆地片,如果是0则表示这块属于海洋,初始化全部为0*/
int area[700]={0};
/***
每个块号面积为多少,记作area[tag_number]=the_area_of_this tag
也可以用容器unordered_map<int,unsigned>area节省空间
***/

举个例子,比如有一海洋为(X,Y),其中上面为陆地编号为1,右面为陆地(编号为1),下面的陆地编号也为1(说明这块土地与(X,Y)上面那块土地相连),左面为海(编号为0)

如下图画的,则area只用加一个标记为1的陆地面积其中一个就行了。

避开重复的数字有很多种方法,如unordered_map,或者建一个大小为4的数组,进行位运算这些都是一种方法,就不一一介绍了。

Solution

/*对代码解释在代码的后面*/
class Solution
{
public:
unsigned tag[50][50] = {0};
int area[700] = {0};
unsigned landNum = 0, X, Y;
void search(const vector<vector<int>> &grid, const unsigned x, const unsigned y)
{
if (grid[x][y] == 1 && tag[x][y] == 0)
{
area[landNum]++;
tag[x][y] = landNum;
/*四个if是边界检查*/
if (x >= 1)
search(grid, x - 1, y);
if (x < X - 1)
search(grid, x + 1, y);
if (y >= 1)
search(grid, x, y - 1);
if (y < Y - 1)
search(grid, x, y + 1);
}
}
int largestIsland(vector<vector<int>> &grid)
{
unsigned maxAnswer = 0, ans;
X = grid.size();
Y = grid[0].size();
for (unsigned i = 0; i < X; i++)
{
for (unsigned j = 0; j < Y; j++)
{
if (grid[i][j] == 1 && tag[i][j] == 0)
{
landNum++;
search(grid, i, j);
}
}
}
/*define the direction*/
enum
{
UP = 0,
RIGHT,
DOWN,
LEFT
};
unsigned tagIsland[4];
unordered_map<unsigned,unsigned> count; for (unsigned i = 0; i < X; i++)
{
for (unsigned j = 0; j < Y; j++)
{
if (grid[i][j] == 0)
{
ans = 1;
tagIsland[UP] = i >= 1 ? tag[i - 1][j] : 0;
tagIsland[RIGHT] = j < Y - 1 ? tag[i][j + 1] : 0;
tagIsland[DOWN] = i < X - 1 ? tag[i + 1][j] : 0;
tagIsland[LEFT] = j >= 1 ? tag[i][j - 1] : 0;
for (unsigned k = UP; k <= LEFT;k++){
if(count[tagIsland[k]]==0&&tagIsland[k]!=0){
ans += area[tagIsland[k]];
count[tagIsland[k]]=1;
}
}
maxAnswer = max(ans, maxAnswer);
count.clear();
}
}
}
if(maxAnswer==0)
return X * Y;
return maxAnswer;
}
};
  1. 我们首先通过search()函数对整个表进行回溯初始化,使所有单独的陆地都被编号。

  2. 然后通过对每块海洋填海后的总area进行判断,保留最大面积maxAnswer,即每次计算出新的area后都有

    maxAnswer=area>maxAnswer?area:maxAnswer;
  3. 在最后return前,如果maxAnswer=0则表示全部都是陆地,可以用反证法验证: 如果至少有一个海洋,maxAnswer至少为1。全是陆地的话只用return整片陆地的总面积即可,即X*Y。

至此,这道题的其中一种做法就完成了。

[Leetcode]827.使用回溯+标记解决最大人工岛问题的更多相关文章

  1. Python基于回溯法解决01背包问题实例

    Python基于回溯法解决01背包问题实例 这篇文章主要介绍了Python基于回溯法解决01背包问题,结合实例形式分析了Python回溯法采用深度优先策略搜索解决01背包问题的相关操作技巧,需要的朋友 ...

  2. Java实现 LeetCode 827 最大人工岛(DFS+暴力模拟)

    827. 最大人工岛 在二维地图上, 0代表海洋, 1代表陆地,我们最多只能将一格 0 海洋变成 1变成陆地. 进行填海之后,地图上最大的岛屿面积是多少?(上.下.左.右四个方向相连的 1 可形成岛屿 ...

  3. 回溯法解决N皇后问题(以四皇后为例)

    以4皇后为例,其他的N皇后问题以此类推.所谓4皇后问题就是求解如何在4×4的棋盘上无冲突的摆放4个皇后棋子.在国际象棋中,皇后的移动方式为横竖交叉的,因此在任意一个皇后所在位置的水平.竖直.以及45度 ...

  4. 用试探回溯法解决N皇后问题

    学校数据结构的课程实验之一. 数据结构:(其实只用了一个二维数组) 算法:深度优先搜索,试探回溯 需求分析: 设计一个在控制台窗口运行的“n皇后问题”解决方案生成器,要求实现以下功能: 由n*n个方块 ...

  5. LeetCode Merge k Sorted Lists 解决报告

    https://oj.leetcode.com/problems/merge-k-sorted-lists/ 归并K已经整理阵列,和分析算法的复杂. 解决报告:无论是不考虑优化,最简单的实现是要重新走 ...

  6. LeetCode编程训练 - 回溯(Backtracking)

    回溯基础 先看一个使用回溯方法求集合子集的例子(78. Subsets),以下代码基本说明了回溯使用的基本框架: //78. Subsets class Solution { private: voi ...

  7. [LeetCode] 827. Making A Large Island

    In a 2D grid of 0s and 1s, we change at most one 0 to a 1. After, what is the size of the largest is ...

  8. 回溯算法——解决n皇后问题

    所谓回溯(backtracking)是通过系统地搜索求解问题的方法.这种方法适用于类似于八皇后这样的问题:求得问题的一个解比较困难,但是检查一个棋局是否构成解很容易. 不多说,放上n皇后的回溯问题代码 ...

  9. [LeetCode] 827. Making A Large Island 建造一个巨大岛屿

    In a 2D grid of 0s and 1s, we change at most one 0 to a 1. After, what is the size of the largest is ...

随机推荐

  1. Java方法区(Method Area)

    方法区与Java堆一样,是各个线程共享的内存区域,他在与存储已被虚拟机加载的类信息,常量,静态变量,即时编译器编译后的代码等数据,虽然Java虚拟机规范把方法区描述为堆得一个逻辑部分,但是他却有一个别 ...

  2. Linux firewalld使用教程+rhce课程实验

    --timeout= 设置规则生效300秒 调试阶段使用,防止规则设置错误导致无法远程连接 实验:在server0机器上部署httpd服务,通过添加富规则,只允许172.25.0.10/32访问,并且 ...

  3. [leetcode]46. Permutations全排列(给定序列无重复元素)

    Given a collection of distinct integers, return all possible permutations. Input: [1,2,3] Output: [ ...

  4. Spring事务管理的四种方式(以银行转账为例)

    Spring事务管理的四种方式(以银行转账为例) 一.事务的作用 将若干的数据库操作作为一个整体控制,一起成功或一起失败.   原子性:指事务是一个不可分割的工作单位,事务中的操作要么都发生,要么都不 ...

  5. Node.js 中使用 ES6 中的 import / export 的方法大全

    https://blog.csdn.net/universsky2015/article/details/83754741

  6. RobotFramework 模拟http接口登录自动化脚本

    RobotFramework 模拟自动化登录脚本思路: 先获取页面cookie值,然后根据cookie值加上请求体提交登录: 一.获取cookie: 以下脚本获取cookie值,并把改脚本封装为关键字 ...

  7. 为什么Firefox在SSH上这么慢?

    为什么Firefox在SSH上这么慢? Modified on: Fri, 13 Jul 2018 18:37:30 +0800 我尝试使用 通过SSH启动Firefox ssh -X user@ho ...

  8. Python之路(第二十八篇) 面向对象进阶:类的装饰器、元类

    一.类的装饰器 类作为一个对象,也可以被装饰. 例子 def wrap(obj): print("装饰器-----") obj.x = 1 obj.y = 3 obj.z = 5 ...

  9. uni-app开发一次,覆盖多端的前端框架

    uni-app 是一个使用 Vue.js 开发跨平台应用的前端框架,开发者编写一套代码,可编译到iOS.Android.H5.小程序等多个平台. 一套代码,运行多个平台 uni-app实现了一套代码, ...

  10. KD-树(下)

    来自于https://zhuanlan.zhihu.com/p/23966698 构造 kd 树的例子 上面抽象的定义和算法确实是很不好理解,举一个例子会清楚很多.首先随机在 中随机生成 13 个点作 ...