这道题是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. asp也玩三层架构(有源代码)

    实体类 <% Class UserInfo Private mintId Public Property Let UserId(intUserId) mintId = intUserId End ...

  2. Android 仿微信朋友圈发表图片拖拽和删除功能

    朋友圈实现原理 我们使用 Android Device Monitor 来分析朋友圈发布图片的界面实现原理.如果需要分析其他应用的界面实现也是采用这种方法哦. 打开 Android Device Mo ...

  3. 【转】Create Hello-JNI with Android Studio

    [转]Create Hello-JNI with Android Studio From:https://codelabs.developers.google.com/codelabs/android ...

  4. HDU 5418 Victor and World (可重复走的TSP问题,状压dp)

    题意: 每个点都可以走多次的TSP问题:有n个点(n<=16),从点1出发,经过其他所有点至少1次,并回到原点1,使得路程最短. 思路: 给了很多重边,选最小的留下即可.任意点可能无法直接到达, ...

  5. SQLITE-更新查询

    SQLite -更新查询 SQLite UPDATE查询用于修改现有表中的记录.您可以使用WHERE子句与更新查询更新选中的行,否则会被更新的所有行. 语法: UPDATE查询的WHERE子句的基本语 ...

  6. UVA10917 A walk trough the Forest (最短路,dp)

    求出家到其他点的最短路径,题目的条件变成了u->v不是回头路等价于d[u]>d[v]. 然后根据这个条件建DAG图,跑dp统计方案数,dp[u] = sum(dp[v]). #includ ...

  7. Scala 的list

    9.1 使用列表 列表类型:跟数组一样,列表也是同质化的(homogeneous).即所有元素都要是同种类型. 列表结构:所有列表由两部分组成:Nil 和 ::(cons). 基本操作:主要有三个:h ...

  8. 开源项目: circular-progress-button

    带进度条显示的按钮, 其效果如下所示: 其由三部分动画组成: 初始状态->圆环状态->完成状态. 0. 实现从初始到圆环的简单实现: 继承自button 类, 设置其背景 public c ...

  9. python读取.mat文件

    可以先看一下.mat中存了些什么: import scipy.io as sio box_file = '/home/bnrc/formatm/test/1479504458876408533_box ...

  10. 企业自颁布服务器证书的有效性验证(C#为例)

    版权声明:本文为博主原创文章,未经博主允许不得转载. https://blog.csdn.net/notjusttech/article/details/72779904 目前根据项目的需要,整理了一 ...