给你一个点数组 points 和一个表示角度的整数 angle ,你的位置是 location ,其中 location = [posx, posy] 且 points[i] = [xi, yi] 都表示 X-Y 平面上的整数坐标。
最开始,你面向东方进行观测。你 不能 进行移动改变位置,但可以通过 自转 调整观测角度。换句话说,posx 和 posy 不能改变。你的视野范围的角度用 angle 表示, 这决定了你观测任意方向时可以多宽。设 d 为你逆时针自转旋转的度数,那么你的视野就是角度范围 [d - angle/2, d + angle/2] 所指示的那片区域。
这题主要确定角的范围,三点确定一个角,确保转后角的范围在视距之内,然后在范围内寻找所有点,边界条件特别多,所以我直接偷了三叶姐的代码……期末考完一定补题!
令 points[i] =(a,b)points[i]=(a,b),与极点关系 dx = a - x;dy = b - ydx=a−x;dy=b−y。
求极角的方式有两种:
使用 atan(\frac{dy}{dx})atan( dx,dy ):值域范围为 [-90°,90°],需要对 dxdx 与 dydy 进行象限讨论,从而将值域范围转化为我们希望的 [0°,360°],同时需要注意 dx = 0dx=0 的边界情况;
使用 atan2(dy, dx)atan2(dy,dx):值域范围为 [-180°,180°],与我们期望的 [0°,360°] 相差一个固定的值,可进行统一转换,也可以直接使用。
class Solution {
double eps = 1e-9;
public int visiblePoints(List<List<Integer>> points, int angle, List<Integer> location) {
int x = location.get(0), y = location.get(1);
List<Double> list = new ArrayList<>();
int cnt = 0;
double pi = Math.PI, t = angle * pi / 180;
for (List<Integer> p : points) {
int a = p.get(0), b = p.get(1);
if (a == x && b == y && ++cnt >= 0) continue;
list.add(Math.atan2(b - y, a - x) + pi);
}
Collections.sort(list);
int n = list.size(), max = 0;
for (int i = 0; i < n; i++) list.add(list.get(i) + 2 * pi);
for (int i = 0, j = 0; j < 2 * n; j++) {
while (i < j && list.get(j) - list.get(i) > t + eps) i++;
max = Math.max(max, j - i + 1);
}
return cnt + max;
}
}