这道题是LeetCode里的第1001道题。

题目要求:

在 N x N 的网格上,每个单元格 (x, y) 上都有一盏灯,其中 0 <= x < N 且 0 <= y < N 。

最初,一定数量的灯是亮着的。lamps[i] 告诉我们亮着的第 i 盏灯的位置。每盏灯都照亮其所在 x 轴、y 轴和两条对角线上的每个正方形(类似于国际象棋中的皇后)。

对于第 i 次查询 queries[i] = (x, y),如果单元格 (x, y) 是被照亮的,则查询结果为 1,否则为 0 。

在每个查询 (x, y) 之后 [按照查询的顺序],我们关闭位于单元格 (x, y) 上或其相邻 8 个方向上(与单元格 (x, y) 共享一个角或边)的任何灯。

返回答案数组 answer。每个值 answer[i] 应等于第 i 次查询 queries[i] 的结果。

示例:

输入:N = 5, lamps = [[0,0],[4,4]], queries = [[1,1],[1,0]]
输出:[1,0]
解释:
在执行第一次查询之前,我们位于 [0, 0] 和 [4, 4] 灯是亮着的。
表示哪些单元格亮起的网格如下所示,其中 [0, 0] 位于左上角:
1 1 1 1 1
1 1 0 0 1
1 0 1 0 1
1 0 0 1 1
1 1 1 1 1
然后,由于单元格 [1, 1] 亮着,第一次查询返回 1。在此查询后,位于 [0,0] 处的灯将关闭,网格现在如下所示:
1 0 0 0 1
0 1 0 0 1
0 0 1 0 1
0 0 0 1 1
1 1 1 1 1
在执行第二次查询之前,我们只有 [4, 4] 处的灯亮着。现在,[1, 0] 处的查询返回 0,因为该单元格不再亮着。

提示:

  1. 1 <= N <= 10^9
  2. 0 <= lamps.length <= 20000
  3. 0 <= queries.length <= 20000
  4. lamps[i].length == queries[i].length == 2

这道题和N皇后问题很像,如果有读者做过与N皇后类似的题且能够熟练使用哈希表,这道题就会迎刃而解,我本人做过N皇后的题目,但不会使用哈希表,通过这次做题,我基本了解了哈希表的使用方法。

好了话不多说,先说说解题思路吧:

最最最重要的是对于每一个要查询的点 (x,y),即为 queries[k][0] 和 queries[k][1],要计算这个点是否有灯光照射到这里,也就是说对于这个点它的上下左右两条线都不能有灯,并且正斜线和逆斜线这两条线上也不能有灯,上下左右很好判断,就是在 lamp 灯中遍历它的 x,y 的值。但是如何去确定斜线方向上的位置是否有灯呢?这就需要基本的数学知识,画图去理解了。

如图,红线是正斜线,线上的点 (x,y) 永远满足 x-y=k(k是定值),黑线逆斜线同理。所以如果要判断所求点 (x,y) 和某一个灯是否在同一斜线上只要去计算他们 x+y 的值或是 x-y 的值是否相等,就行了。代码如下:

解题代码:

C++ 使用的是 vector:

class Solution {
public:
vector<int> gridIllumination(int N, vector<vector<int>>& lamps, vector<vector<int>>& queries) {
vector<int>res;
int light[lamps.size()]={0};//定义一个数组来查询等是否关闭,0代表开,1代表关
//for(int i=0;i<lamps.size();i++)
// cout<<light[i]<<endl;
int state;//保存查询的点是亮的还是不亮的的状态
for(int i=0;i<queries.size();i++){
state=0;
long long qx=queries[i][0];//为了保证不会数据溢出
long long qy=queries[i][1];
for(int j=0;j<lamps.size();j++){
if(light[j]==1)continue;//如果灯关了,就不用继续判断
long long lx=lamps[j][0];//为了保证不会数据溢出
long long ly=lamps[j][1];
if(qx==lx||qy==ly||qx-qy==lx-ly||qx+qy==lx+ly){
state=1;
if((abs((qx-lx))<=1||abs((qy-ly))<=1)&&//保证不会超时
(qx-lx)*(qx-lx)+(qy-ly)*(qy-ly)<=2)
light[j]=1;//把四周离点距离小于等于根号2的灯关闭
}
}
res.push_back(state);
}
return res;
}
};

Java 使用的是哈希表:

import java.util.Map;
import java.util.Set;
import java.util.HashMap;
import java.util.HashSet; class Solution {
private static Map<Integer,Set<String>>lamp_X;
private static Map<Integer,Set<String>>lamp_Y;
private static Map<Integer,Set<String>>lamp_P;
private static Map<Integer,Set<String>>lamp_R; public static int[] gridIllumination(int N, int[][] lamps, int[][] queries) {
int[]res = new int[queries.length];
initialization(lamps); for(int i = 0;i < queries.length;i++) {
int X = queries[i][0];
int Y = queries[i][1]; int state = isLight(X,Y)?1:0;
res[i] = state;
if(state == 1) {//依次关灯
turnOff(X,Y);
turnOff(X-1,Y);
turnOff(X+1,Y);
turnOff(X,Y+1);
turnOff(X,Y-1);
turnOff(X-1,Y+1);
turnOff(X+1,Y+1);
turnOff(X-1,Y-1);
turnOff(X+1,Y-1);
}
}
return res;
} public static void turnOff(int X,int Y) {
int P = X + Y;
int R = X - Y; StringBuilder sb = new StringBuilder();
sb.append(X).append(',').append(Y);
//关灯
if(lamp_X.containsKey(X)) {
lamp_X.get(X).remove(sb.toString());
}
if(lamp_Y.containsKey(Y)) {
lamp_Y.get(Y).remove(sb.toString());
}
if(lamp_P.containsKey(P)) {
lamp_P.get(P).remove(sb.toString());
}
if(lamp_R.containsKey(R)) {
lamp_R.get(R).remove(sb.toString());
}
} public static boolean isLight(int X,int Y) {
int P = X + Y;
int R = X - Y;
//判断是否有灯,就是看是否有键,而且要满足键有一个非空集合的映射
if(lamp_X.containsKey(X) && !lamp_X.get(X).isEmpty())
return true;
if(lamp_Y.containsKey(Y) && !lamp_Y.get(Y).isEmpty())
return true;
if(lamp_P.containsKey(P) && !lamp_P.get(P).isEmpty())
return true;
if(lamp_R.containsKey(R) && !lamp_R.get(R).isEmpty())
return true;
return false;
} public static void initialization(int[][] lamps) {
lamp_X = new HashMap<>(); //横向
lamp_Y = new HashMap<>(); //纵向
lamp_P = new HashMap<>(); //正斜向
lamp_R = new HashMap<>(); //逆斜向
StringBuilder sb = new StringBuilder(); for(int[] i : lamps) {
int X = i[0];
int Y = i[1];
int P = X + Y;
int R = X - Y;
//首先要知道是否有X的键,没有则需要创建一个映射
if(!lamp_X.containsKey(X)) {
lamp_X.put(X, new HashSet<>());
}
if(!lamp_Y.containsKey(Y)) {
lamp_Y.put(Y, new HashSet<>());
}
if(!lamp_P.containsKey(P)) {
lamp_P.put(P, new HashSet<>());
}
if(!lamp_R.containsKey(R)) {
lamp_R.put(R, new HashSet<>());
} sb.setLength(0);//这里相当于初始化,清除上一次的点位
sb.append(X).append(',').append(Y);
//加入集合中
lamp_X.get(X).add(sb.toString());
lamp_Y.get(Y).add(sb.toString());
lamp_P.get(P).add(sb.toString());
lamp_R.get(R).add(sb.toString());
}
}
}

提交结果:

C++的结果:

Java的结果:

个人总结:

通过本次解题的过程,我对 Java 的 Map,Set,HashMap,HashSet 接口和类都有了一定的了解,同时也复习了一下 C++ 的写法,感觉哈希表真的是一种高效的算法。

【LeetCode】Grid Illumination(网格照明)的更多相关文章

  1. SilverLight:布局(1) Border(边框)对象、Grid(网格)对象

    ylbtech-SilverLight-Layout:布局(1) Border(边框)对象.Grid(网格)对象 A, Border(边框)对象 B, Grid(网格)对象 C, Grid(网格)对象 ...

  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. 【LeetCode】1001. Grid Illumination 解题报告(C++)

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

  4. 【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 ...

  5. CSS Grid基于网格的二维布局系统(详细教程)

    .grid-wrap{ display: inline-flex; padding: 20px; background: #f4f4f4; word-break: initial; } .handle ...

  6. grid - 通过网格区域命名和定位网格项目

    1.像网格线名称一样,网格区域的名称也可以使用grid-template-areas属性来命名.引用网格区域名称也可以设置网格项目位置. 设置网格区域的名称应该放置在单引号或双引号内,每个名称由一个空 ...

  7. grid - 重复网格轨道

    使用repeat()可以创建重复的网格轨道.这个适用于创建相等尺寸的网格项目和多个网格项目 <view class="grid"> <view class='gr ...

  8. 网页布局——grid弹性网格布局

    网格布局(Grid)是最强大的 CSS 布局方案. Flexbox 是为一维布局设计的,而 Grid 是为二维布局设计. grid目前兼容性目前还可以,主流浏览器对它的支持力度很大,ie9,10宣布它 ...

  9. CSS--使用CSS Grid(网格)布局

    .一 CSS Grid(网格布局)简介 CSS Grid 布局由两个核心组成部分是父元素和子元素,父元素 是实际的 grid(网格),子元素是 grid(网格) 内的内容.下面是一个父元素和六个子元素 ...

随机推荐

  1. 编写可执行程序,其它程序调用,并返回数据,C#

    有时候在创建临时文件或文件夹,使用完成后,释放失败,删除提示占用,又不能结束主程序,就可以通过别的方法来解决 比如,另外创建一个程序,单独执行任务,完成后结束程序,并返回执行结果,上述问题就能解决. ...

  2. ssh框架出现Java.lang.NoSuchMethodError: antlr.collections.AST.getLine()I错误

    原因:因为Struts自带的antlr-2.7.2.jar,比Hibernate自带的antlr-2.7.7.jar的版本要低,存在jar包冲突现象,因此要删除前一个低版本的. 由于myeclipse ...

  3. DVWA之命令注入(command injection)

    Command injection就是指通过提交恶意构造的参数破坏命令语句结构,从而达到执行恶意命令的目的 LOW 无论是Windows还是Linux,都可以使用&&连接多个命令 执行 ...

  4. sqlite的应用

    对于Android平台来说,系统内置了丰富的API来供开发人员操作SQLite,我们可以轻松的完成对数据的存取.下面就向大家介绍一下SQLite常用的操作方法.本篇文章主要用到SQLiteDataba ...

  5. uvm_reg_model——寄存器模型(一)

    对于一个复杂设计,寄存器模型要能够模拟任意数量的寄存器域操作.UVM提供标准的基类库,UVM的寄存器模型来自于继承自VMM的RAL(Register Abstract Layer),现在可以先将寄存器 ...

  6. C#遍历文件夹下全部文件

    public static List<string> GetFile(string path, List<string> FileList, string RelativePa ...

  7. 允许Java App(applet)粘贴方法

    修改安全策略文件: "java.policy" JRE6的路径在:"C:\Program Files (x86)\Java\jre6\lib\security" ...

  8. (七)VMware Harbor 问题:Get https://192.168.3.135:8088/v2/: http:server gave HTTP response to HTTPS client

    (一)问题描述 登陆时,报错 docker Get https://192.168.3.135:8088/v2/: http:server gave HTTP response to HTTPS cl ...

  9. python基础一 day13 迭代器

    # 双下方法# print([1].__add__([2]))# print([1]+[2]) # 迭代器# l = [1,2,3]# 索引# 循环 for# for i in l:# i## for ...

  10. nodeJS进程管理器pm2

    pm2是一个带有负载均衡功能的Node应用的进程管理器.当你要把你的独立代码利用全部的服务器上的所有CPU,并保证进程永远都活着,0秒的重载, PM2是完美的. PM2是开源的基于Nodejs的进程管 ...