来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/grid-illumination

题目描述

在大小为 n x n 的网格 grid 上,每个单元格都有一盏灯,最初灯都处于 关闭 状态。

给你一个由灯的位置组成的二维数组 lamps ,其中 lamps[i] = [rowi, coli] 表示 打开 位于 grid[rowi][coli] 的灯。即便同一盏灯可能在 lamps 中多次列出,不会影响这盏灯处于 打开 状态。

当一盏灯处于打开状态,它将会照亮 自身所在单元格 以及同一 行 、同一 列 和两条 对角线 上的 所有其他单元格 。

另给你一个二维数组 queries ,其中 queries[j] = [rowj, colj] 。对于第 j 个查询,如果单元格 [rowj, colj] 是被照亮的,则查询结果为 1 ,否则为 0 。在第 j 次查询之后 [按照查询的顺序] ,关闭 位于单元格 grid[rowj][colj] 上及相邻 8 个方向上(与单元格 grid[rowi][coli] 共享角或边)的任何灯。

返回一个整数数组 ans 作为答案, ans[j] 应等于第 j 次查询 queries[j] 的结果,1 表示照亮,0 表示未照亮。

示例 1:

输入:n = 5, lamps = [[0,0],[4,4]], queries = [[1,1],[1,0]]
输出:[1,0]
解释:最初所有灯都是关闭的。在执行查询之前,打开位于 [0, 0] 和 [4, 4] 的灯。

第 0 次查询检查 grid[1][1] 是否被照亮(蓝色方框)。该单元格被照亮,所以 ans[0] = 1 。然后,关闭红色方框中的所有灯。

第 1 次查询检查 grid[1][0] 是否被照亮(蓝色方框)。该单元格没有被照亮,所以 ans[1] = 0 。然后,关闭红色矩形中的所有灯。

示例 2:

输入:n = 5, lamps = [[0,0],[4,4]], queries = [[1,1],[1,1]]
输出:[1,1]

示例 3:

输入:n = 5, lamps = [[0,0],[0,4]], queries = [[0,4],[0,1],[1,4]]
输出:[1,1,0]

提示:

1 <= n <= 109
0 <= lamps.length <= 20000
0 <= queries.length <= 20000
lamps[i].length == 2
0 <= rowi, coli < n
queries[j].length == 2
0 <= rowj, colj < n

解题思路

看题目难度hard和样本规模109基本就放弃了暴力模拟的方法。这道题有两个难点,一个是灯是如何表示亮暗,一个是位置亮暗如何判断。

首先由于n的规模巨大,所以我们不可能用n*n的矩阵来存储每一个灯的状态,所以我们仅仅只能记录亮着的灯的位置,而由于要频繁的查询和删除晾着的灯,我们使用一个set来表示亮着的灯,将二维坐标转换为一维坐标,使用row * n + col 来表示灯的一维坐标。这里有个坑点就是由于row和col的最大值很大,所以row*n+col可能超出int的最大表示范围,所以转换为long long。

再来考虑如何判断查询点是否亮暗,使用四个map做哈希表来分别存储亮暗的状态,使用row来存储行亮的状态,row的键就是代表行数,值代表此行有几盏亮着的灯。col的键代表列数,值代表此行有几盏亮着的等。left表示左上到右下的对角线,它的键是表示对角线与竖边界的交点的行数,也就是row - col 值代表着此对角线有几盏两者的灯,right和left对角线类似。这样我们就可以将整个网格的亮的状态用这四个哈希表表示出来,查询是时候也只需要查询查询点所在行列对角线是否有亮着的灯就好了。需要注意的是,在亮灯的数据中有重复打开同一盏灯的操作,需要排除。

查询结束后,根据查询点将九个位置的灯灭掉,更新四个哈希表的值即可,在灭九个位置灯的过程中,要考虑坐标是否已经超出网格边界。

代码展示

class Solution {
public:
vector<int> gridIllumination(int n, vector<vector<int>>& lamps, vector<vector<int>>& queries) {
vector<int> viRet;
set<long long> setiLamps;
unordered_map<int, int> mapiiRow, mapiiCol, mapiiLeft, mapiiRight;
for(auto iter: lamps)
{
long long temp = (long long)iter[0] * n + iter[1];
auto setIter = setiLamps.find(temp);
if(setIter == setiLamps.end())
{
setiLamps.insert(temp);
mapiiRow[iter[0]]++;
mapiiCol[iter[1]]++;
mapiiLeft[iter[0] - iter[1]]++;
mapiiRight[iter[0] + iter[1]]++;
}
}
for(auto iter: queries)
{
if(mapiiRow[iter[0]] || mapiiCol[iter[1]] || mapiiLeft[iter[0] - iter[1]] || mapiiRight[iter[0] + iter[1]])
{
viRet.push_back(1);
}
else
{
viRet.push_back(0);
}
for(int i = -1; i < 2; i++)
{
for(int j = -1; j < 2; j++)
{
int iRow = iter[0] + i, iCol = iter[1] + j;
if(iRow < 0 || iRow >= n || iCol < 0 || iCol >= n)
continue;
long long temp = (long long)iRow * n + iCol;
auto setIter = setiLamps.find(temp);
if(setIter != setiLamps.end())
{
mapiiRow[iRow]--;
mapiiCol[iCol]--;
mapiiLeft[iRow - iCol]--;
mapiiRight[iRow + iCol]--;
setiLamps.erase(setIter);
}
}
}
}
return viRet;
}
};

运行结果

LeetCode-1001 网格照明的更多相关文章

  1. 【LeetCode】Grid Illumination(网格照明)

    这道题是LeetCode里的第1001道题. 题目要求: 在 N x N 的网格上,每个单元格 (x, y) 上都有一盏灯,其中 0 <= x < N 且 0 <= y < N ...

  2. [Swift]LeetCode1001. 网格照明 | Grid Illumination

    On a N x N grid of cells, each cell (x, y) with 0 <= x < N and 0 <= y < N has a lamp. In ...

  3. Swift LeetCode 目录 | Catalog

    请点击页面左上角 -> Fork me on Github 或直接访问本项目Github地址:LeetCode Solution by Swift    说明:题目中含有$符号则为付费题目. 如 ...

  4. LeetCode刷题总结-哈希表篇

    本文总结在LeetCode上有关哈希表的算法题,推荐刷题总数为12题.具体考察的知识点如下图: 1.数学问题 题号:149. 直线上最多的点数,难度困难 题号:554. 砖墙,难度中等(最大最小边界问 ...

  5. leetcode难题

    4 寻找两个有序数组的中位数       35.9% 困难     10 正则表达式匹配       24.6% 困难     23 合并K个排序链表       47.4% 困难     25 K ...

  6. 【Leetcode周赛】从contest-121开始。(一般是10个contest写一篇文章)

    Contest 121 (题号981-984)(2019年1月27日) 链接:https://leetcode.com/contest/weekly-contest-121 总结:2019年2月22日 ...

  7. LeetCode:Minimum Path Sum(网格最大路径和)

    题目链接 Given a m x n grid filled with non-negative numbers, find a path from top left to bottom right ...

  8. [LeetCode] Magic Squares In Grid 网格中的神奇正方形

    A 3 x 3 magic square is a 3 x 3 grid filled with distinct numbers from 1 to 9 such that each row, co ...

  9. 【LeetCode】1001. Grid Illumination 解题报告(C++)

    作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 哈希 日期 题目地址:https://leetcod ...

  10. 【leetcode】1001. Grid Illumination

    题目如下: On a N x N grid of cells, each cell (x, y) with 0 <= x < N and 0 <= y < N has a la ...

随机推荐

  1. 使用sanic框架实现分布式爬虫

    bee_server.py from sanic import Sanic from sanic import response from urlpool import UrlPool #初始化url ...

  2. gulp报错The following tasks did not complete

    代码如下: //引用gulp模块 const gulp = require('gulp'); //使用gulp.task()建立任务 gulp.task('first', () => { con ...

  3. Kafka技术专题之「性能调优篇」消息队列服务端出现内存溢出OOM以及相关性能调优实战分析

    内存问题 本篇文章介绍Kafka处理大文件出现内存溢出 java.lang.OutOfMemoryError: Direct buffer memory,主要内容包括基础应用.实用技巧.原理机制等方面 ...

  4. 使用linux的ffmpeg进行B站直播推流

    很久之前买了个友善的开发板R2S,一直在家吃灰.最近看到网上有用ffmpeg进行直播推流的案例,想把吃灰的的开发板利用起来,于是有了这篇教程. 第一步:安装ffmpeg sudo apt update ...

  5. 【转载】SQL SERVER 表变量与临时表的优缺点

    什么情况下使用表变量?什么情况下使用临时表? -- 表变量: DECLARE @tb table(id int identity(1,1), name varchar(100)) INSERT @tb ...

  6. 在生产中部署 ES2015+ 代码

    大多数 Web 开发人员都喜欢编写具有所有最新语言特性的 JavaScript--async/await.类.箭头函数等.然而,尽管事实上所有现代浏览器都可以运行 ES2015+ 代码并原生支持我刚才 ...

  7. 3D视觉算法初学概述

    背景知识 RGB-D相机 一,基于3DMM的三维人脸重建技术概述 1.1,3D 人脸重建概述 1.2,初版 3DMM 二,视觉SLAM算法基础概述 2.1,视觉里程计 2.2,后端优化 2.3,回环检 ...

  8. [C++]我的理解之内存对齐

    问题1:为什么要内存对齐? 平台原因:不是所有的平台都能访问到任意地址上的任何数据,如果在特定的地址上找不到数据的话就会抛出硬件异常. 性能问题:简单的来说如果没有使用内存对齐的话,相对于内存对齐,C ...

  9. day05-Spring管理Bean-IOC-03

    Spring管理Bean-IOC-03 2.基于XML配置bean 2.15bean的生命周期 bean对象的创建是由JVM完成的,然后执行如下方法: 执行构造器 执行set相关方法 调用bean的初 ...

  10. chrome实现下载文件JS代码弹出'另存为'窗口

    1.TXT类型文件 <!DOCTYPE html> <html lang="en"> <head> <meta charset=" ...