【LeetCode】Grid Illumination(网格照明)
这道题是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 <= N <= 10^90 <= lamps.length <= 200000 <= queries.length <= 20000lamps[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(网格照明)的更多相关文章
- SilverLight:布局(1) Border(边框)对象、Grid(网格)对象
ylbtech-SilverLight-Layout:布局(1) Border(边框)对象.Grid(网格)对象 A, Border(边框)对象 B, Grid(网格)对象 C, Grid(网格)对象 ...
- [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 ...
- 【LeetCode】1001. Grid Illumination 解题报告(C++)
作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 哈希 日期 题目地址:https://leetcod ...
- 【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 ...
- CSS Grid基于网格的二维布局系统(详细教程)
.grid-wrap{ display: inline-flex; padding: 20px; background: #f4f4f4; word-break: initial; } .handle ...
- grid - 通过网格区域命名和定位网格项目
1.像网格线名称一样,网格区域的名称也可以使用grid-template-areas属性来命名.引用网格区域名称也可以设置网格项目位置. 设置网格区域的名称应该放置在单引号或双引号内,每个名称由一个空 ...
- grid - 重复网格轨道
使用repeat()可以创建重复的网格轨道.这个适用于创建相等尺寸的网格项目和多个网格项目 <view class="grid"> <view class='gr ...
- 网页布局——grid弹性网格布局
网格布局(Grid)是最强大的 CSS 布局方案. Flexbox 是为一维布局设计的,而 Grid 是为二维布局设计. grid目前兼容性目前还可以,主流浏览器对它的支持力度很大,ie9,10宣布它 ...
- CSS--使用CSS Grid(网格)布局
.一 CSS Grid(网格布局)简介 CSS Grid 布局由两个核心组成部分是父元素和子元素,父元素 是实际的 grid(网格),子元素是 grid(网格) 内的内容.下面是一个父元素和六个子元素 ...
随机推荐
- Linux下自动化测试环境的搭建
1.安装Linux虚拟机,详情参考 https://blog.csdn.net/qq_22770715/article/details/78558374 https://www.cnblog ...
- Kettle Rest大文件上传(RestUploadFile.ktr) Rest文件下载(FileDownload.ktr)
1. Rest大文件上传(RestUploadFile.ktr) 需求描述 上传文件大于10M小于500M 上传文件进行分片(5M一片要比1M分片整体时间快) 先使用java类进行功能模拟在迁移Ktr ...
- CentOS7下Oracle11gR2监听启动错误解决
oracle监听程序启动失败,错误如下: [oracle@localhost ~]$ lsnrctl start LSNRCTL - Production on -APR- :: Copyright ...
- 基于Myeclipse的三大框架(SSH)整合
文中主要基于Myeclipse进行配置,配置流程为:Hibernate --> Spring --> 整合 --> struts2 -->整合.注意:在此文中,主要讲述基于注解 ...
- 安装MySQLdb出现HAVE_WCSCOLL重定义问题的解决方法
root@wodeyitian MySQL-python-1.2.3]# python setup.py install running install running bdist_egg runni ...
- webpack前端构建工具学习总结(四)之自动化生成项目中的html页面
接续上文:webpack前端构建工具学习总结(三)之webpack.config.js配置文件 插件的介绍文档:https://www.npmjs.com/package/html-webpack-p ...
- Software Engineer(百赴美)
http://talent.baidu.com/component1000/corp/baidu/html/BFM.html http://talent.baidu.com/baidu/web/tem ...
- python之常见的坑
li = [1,2,3,4] # [1,3,4] # 索引值是奇数的删除 for i in range(4): if i % 2 == 1: li.pop(i) # 会报错 print(li) 面试题 ...
- 利用VS自带的命令行工具查看和生产PublicKeyToken
使用VS2008(或其他版本)命令行工具,键入:SN -T C:\*****.dll 就会显示出该dll具体的PublicKeyToken数值. 如果该程序集没有强命 名,则不会有PublicKeyT ...
- iOS Crash
常见原因及解决方法: 1. 访问数组类对象越界或插入了空对象NSMutableArray/NSMutableDictionary/NSMutableSet 等类下标越界,或者 insert 了一个 n ...