这道题是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. Oracle 恢复数据后,数据库中中文变成问号解决方法

    1.右击---我的电脑---环境变量 2.新增环境变量 变量名:LANG=zh_CN.GBK NLS_LANG=SIMPLIFIED CHINESE_CHINA.ZHS16GBK 3.重启PLSQL或 ...

  2. Java虚拟机(JVM),JDK,JRE和JVM的区别——通过示例学习Java编程(2)

    Java虚拟机(JVM),JDK,JRE和JVM的区别 作者:CHAITANYA SINGH 来源:https://www.koofun.com/pro/kfpostsdetail?kfpostsid ...

  3. 用Excel生成Sql

    用Excel生成Sql: 以如图为例:点击一行数据的后面一个单元格,在上面的fx部分输入=,以等号开头证明这是一个公式.在等号的后面写上想要添加的数据,书写规范是这样:'"&A2&a ...

  4. this/super/static/final/匿名对象/继承/抽象类/访问权限修饰符

    1.this关键字的作用     1)调用本类中的属性;     2)调用本类中的构造方法;且只能放首行,且必须留一个构造方法作为出口,即不能递归调用     3)表示当前对象; 2.匿名对象     ...

  5. svn merge当主干修改后合并分支

    例如版本r1的主干创建分支r2,在r2上修改后得到r3,r1之后也修改得到r4,现在合并分支到主干上: 如果r3的修改和r4有冲突会提示出现冲突,因此不用担心主干合并后会被分支操作覆盖,因为这并不是简 ...

  6. Windows Server 2008 R2中上传和下载文件

    在 Windows Server 2008 R2 中,使用服务器管理器来启用或禁用 Windows 功能,那在这个上面如何上传和下载文件呢? 1.在“服务器管理器”->“角色”->“web ...

  7. Cordova插件中JavaScript代码与Java的交互细节介绍

    在Cordova官网中有这么一张架构图:大家看右下角蓝色的矩形框"Custom Plugin"--自定义插件.意思就是如果您用Cordova打包Mobile应用时,发现您的移动应用 ...

  8. 判断NumLock键和CapsLock键是否被锁定

    实现效果: 知识运用: AIP函数GetKeyState //针对已处理过的按键 在最近一次输入信息时 判断指定虚拟键的状态 intkey:预测试的虚拟键键码 实现代码: [DllImport(&qu ...

  9. CPP-基础:strcpy之于C++(

    以下对strcpy函数错误的是? char atr1[]="string"; ]; char *str3; char *str4="sting"; A.strc ...

  10. shell脚本,按字母出现频率降序排序。

    [root@localhost oldboy]# cat file the squid project provides a number of resources toassist users de ...