1610. Maximum Number of Visible Points
Problem Description
In this problem, we are given a list of points on a 2-D plane, an angle, and a location which represents our position. Our initial view direction faces due east, and we cannot physically move from our position, but we can rotate to change the field of view. The angle
given signifies the total span of our field of view in degrees, meaning we can see points that lie within an angular range of angle
degrees from our current direction of sight.
Every point is defined by coordinates [x_i, y_i]
, and our location is [pos_x, pos_y]
. The key task is to find out the maximum number of points we can see from our location when choosing the optimal rotation. Points located at our position are always visible, and the visibility of other points is determined by whether the angle formed by a point, our position, and the immediate east direction from our position is within our field of view.
Moreover, multiple points may share the same coordinates, and there can also be points at our current location. Importantly, the presence of points does not block our view of other points.
Intuition
To solve the problem, we use the concept of angular sweep. We calculate the angle between the east direction and the line connecting our location to each point. If there's a point at our location, it's always visible, so we count these points directly by incrementing a counter each time we encounter such a point.
For the remaining points, we convert their coordinates into angles using the atan2
function, which handles the correct quadrant of the angle for us. Then we sort these angles in ascending order for a sweeping process.
Afterward, we replicate the sorted angles while adding 2 * pi
to each, effectively simulating a circular sweep beyond the initial 0 to 2*pi range to handle the wrap-around at the 360-degree mark. This way, we can perform a linear sweep using a two-pointer technique or binary search.
We iterate through each angle, considering it as the left boundary of our field of view, and then find using binary search (via bisect_right
) the farthest angle that can fit within the field of view spanned by angle
degrees. By doing this for all starting angles, and finding the maximum range of points that can fit within our field of view for each start angle, we can establish the maximum number of points visible after the optimal rotation.
We then add the count of points at our location to this maximum number to get the total maximum number of points we can see. This is because points at our location are not subject to angular constraints and are always counted.
Learn more about Math, Sorting and Sliding Window patterns.
Solution Approach
The solution follows a specific pattern of dealing with geometric problems involving an angular field of view:
-
Calculate Angles: We start by calculating the angles for all the points relative to our position and the east direction. We use the function
atan2(yi - y, xi - x)
to calculate the angle of each point relative to our location,location = [x, y]
. Theatan2
function helps in getting the angle from the x-axis to the point(xi, yi)
which takes into account the correct quadrant of the point. -
Count Overlapping Points: We maintain a counter,
same
, to count the number of points that exactly overlap with our current location as these points are always visible. -
Sort Angles: We sort the list of angles to prepare for the sweeping process. Since we'll be doing an angular sweep, a sorted array of angles is essential.
-
Extend Angles: In order to smoothly handle the wrap-around at 360 degrees (or
2 * pi
radians), we extend our list of anglesv
by appending each angle incremented by2 * pi
. This is done byv += [deg + 2 * pi for deg in v]
, effectively doubling the interval of the angle list to simulate a circular region. -
Sliding Window / Binary Search: We apply the concept of a sliding window or a binary search to find the maximal number of points that fit within any
angle
span:- Using a for loop, iterate over each angle in the original list
v
(not the extended list), treating it as the start of the field of view. - For each starting angle, use
bisect_right
from thebisect
module to conduct a binary search to find how far (to the right in the sorted list) you can go before exceeding the boundary of your field of view. The endpoint isv[i] + t
wheret
is theangle
converted to radians (angle * pi / 180
). - The
bisect_right
function will return the index of the farthest angle that can be included which, when subtracted by the starting indexi
, gives us the number of points within this particular field of view window.
- Using a for loop, iterate over each angle in the original list
-
Find Maximum: The maximum number of points within any such window across all possible starting points is found by
mx = max((bisect_right(v, v[i] + t) - i for i in range(n)), default=0)
. This takes the highest number from all the window sweeps we performed. -
Add Overlapping Points: Finally, we add the
same
counter value to ourmx
to count those points that overlap with our position, yielding the final answer:return mx + same
.
By wrapping up all of these steps inside the visiblePoints
function, we get a robust solution that is capable of returning the maximum number of points that can be seen by rotating to the optimal angle.
Ready to land your dream job?
Unlock your dream job with a 2-minute evaluator for a personalized learning plan!
Start EvaluatorExample Walkthrough
Let's go through a small example to illustrate the solution approach. Imagine you are located at [0, 0]
(the origin), and you have an angle
of 90 degrees for your field of view. This means you can see all points that are within 90 degrees to your east.
Consider the following list of points:
points = [[-1,0], [1,0], [0,0], [0,1], [1,1]]
We will use the steps from the solution approach to find the maximum number of points visible:
-
Calculate Angles:
- We calculate the angles for
[-1,0]
,[1,0]
,[0,1]
, and[1,1]
because[0,0]
is the location itself. - For point
[-1,0]
,atan2(0, -1)
gives us an angle of π (180 degrees). - For point
[1,0]
,atan2(0, 1)
gives us an angle of 0 degrees (due east). - For point
[0,1]
,atan2(1, 0)
gives us an angle of π/2 (90 degrees). - For point
[1,1]
,atan2(1, 1)
gives us an angle of π/4 (45 degrees).
- We calculate the angles for
-
Count Overlapping Points:
- The
same
counter is 1 because one point overlaps with our position[0,0]
.
- The
-
Sort Angles:
- We sort the angles, not including the point at the origin:
[0, π/4, π/2, π]
.
- We sort the angles, not including the point at the origin:
-
Extend Angles:
- The angle list is extended to handle the 360-degree wrap-around. It becomes
[0, π/4, π/2, π, 2π, 9π/4, 5π/2, 3π]
.
- The angle list is extended to handle the 360-degree wrap-around. It becomes
-
Sliding Window / Binary Search:
- We use our field of view angle (90 degrees = π/2 radians) as the window size.
- For angle
0
, we can see up to π/4 before the field of view ends (covering points[1,0]
and[1,1]
). - For angle π/4, we can see up to 9π/4 (also covering
[1,0]
and[1,1]
). - For angle π/2, we can still see up to 9π/4 (covering only
[0,1]
). - For angle π, the visible range is beyond our extended list, so it doesn't cover any extra point in the original list.
-
Find Maximum:
- The maximum number of points visible in any window is 2 (from angles
0
andπ/4
).
- The maximum number of points visible in any window is 2 (from angles
-
Add Overlapping Points:
- We add the
same
counter which is 1, so the final maximum number of points visible is2 + 1 = 3
.
- We add the
Hence, for this example, rotating to face any angle between due east (0 degrees) and 45 degrees would allow us to see three points: the one at our position and the points at [1,0]
and [1,1]
.
Solution Implementation
1from math import atan2, pi
2from bisect import bisect_right
3from typing import List
4
5class Solution:
6 def visiblePoints(self, points: List[List[int]], angle: int, location: List[int]) -> int:
7 # Stores the polar angles of points with respect to the location
8 polar_angles = []
9
10 # Coordinates of the location from which we are looking
11 location_x, location_y = location
12
13 # Counter for points at the same location as the observer
14 overlap_count = 0
15
16 # Calculate polar angles of points relative to the observer's location
17 for point_x, point_y in points:
18 # If the point is at the observer's location, increment overlap count
19 if point_x == location_x and point_y == location_y:
20 overlap_count += 1
21 else:
22 # Compute the polar angle and add it to the list
23 angle = atan2(point_y - location_y, point_x - location_x)
24 polar_angles.append(angle)
25
26 # Sort the angles to prepare for angle-range search
27 polar_angles.sort()
28
29 # Number of unique points (excluding overlaps)
30 num_points = len(polar_angles)
31
32 # Append mirrored polar angles to simulate a circular sweep
33 polar_angles += [angle + 2 * pi for angle in polar_angles]
34
35 # Convert the viewing angle to radians for comparison
36 max_radians = angle * pi / 180
37
38 # Use a sliding window approach to find the maximum number of points in any angle-range
39 # We use bisect_right to find the rightmost position to which max_radians can be added
40 max_visible = max(
41 (bisect_right(polar_angles, polar_angles[i] + max_radians) - i for i in range(num_points)),
42 default=0,
43 )
44
45 # Return the sum of the maximum visible points and any overlapping points
46 return max_visible + overlap_count
47
1class Solution {
2 public int visiblePoints(List<List<Integer>> points, int angle, List<Integer> location) {
3 // List to store the angles from the observer's location to each point
4 List<Double> angles = new ArrayList<>();
5 int observerX = location.get(0), observerY = location.get(1);
6 int overlapCount = 0; // Count of points overlapping the observer's location
7
8 // Iterate over all points to calculate angles or count overlaps
9 for (List<Integer> point : points) {
10 int pointX = point.get(0), pointY = point.get(1);
11
12 // Check if the current point overlaps with the observer's location
13 if (pointX == observerX && pointY == observerY) {
14 overlapCount++; // Increment overlap count
15 continue; // Skip the rest of the loop
16 }
17
18 // Calculate the angle and add it to the list
19 angles.add(Math.atan2(pointY - observerY, pointX - observerX));
20 }
21
22 // Sort the angles in ascending order
23 Collections.sort(angles);
24
25 // Duplicate the angles list to handle the circular nature of angles
26 int anglesCount = angles.size();
27 for (int i = 0; i < anglesCount; ++i) {
28 angles.add(angles.get(i) + 2 * Math.PI);
29 }
30
31 // Convert the viewing angle to radians
32 double threshold = angle * Math.PI / 180;
33
34 // Initialize the maximum number of visible points
35 int maxVisible = 0;
36
37 // Two-pointer technique to find the maximum number of points visible within the angle range
38 for (int left = 0, right = 0; right < 2 * anglesCount; ++right) {
39 // Shift the left pointer to the right until the points are outside the viewing angle
40 while (left < right && angles.get(right) - angles.get(left) > threshold) {
41 left++; // Narrow the range by moving the left pointer to the right
42 }
43
44 // Update maxVisible with the maximum number of points in the current range
45 maxVisible = Math.max(maxVisible, right - left + 1);
46 }
47
48 // Return the maximum number of visible points plus any overlapping points
49 return maxVisible + overlapCount;
50 }
51}
52
1#include <vector>
2#include <algorithm>
3#include <cmath>
4
5class Solution {
6public:
7 int visiblePoints(std::vector<std::vector<int>>& points, int angle, std::vector<int>& location) {
8 std::vector<double> angles; // Vector to store the angles of the visible points
9 int observer_x = location[0], observer_y = location[1]; // Observer's location
10 int overlappingPoints = 0; // Count of points that overlap with the observer's location
11 for (const auto& point : points) {
12 int point_x = point[0], point_y = point[1];
13 if (point_x == observer_x && point_y == observer_y) {
14 // Increment count if the point's location is the same as the observer's location
15 ++overlappingPoints;
16 } else {
17 // Calculate and store the angle from the observer to the point
18 angles.push_back(atan2(static_cast<double>(point_y - observer_y),
19 static_cast<double>(point_x - observer_x)));
20 }
21 }
22
23 // Sort the angles in ascending order
24 std::sort(angles.begin(), angles.end());
25
26 int totalAngles = angles.size(); // Total number of unique angles
27 // Duplicate the angles by appending 2*PI to each to handle the circular range
28 for (int i = 0; i < totalAngles; ++i) {
29 angles.push_back(angles[i] + 2 * M_PI);
30 }
31
32 int maxVisible = 0; // Maximum number of points visible within the angle
33 double radianAngle = angle * M_PI / 180; // Convert angle to radians
34
35 // Two-pointer approach to find the max number of points that fit within the angle
36 for (int left = 0, right = 0; right < 2 * totalAngles; ++right) {
37 // Move the left pointer until the points no longer fit in the viewing angle
38 while (left < right && angles[right] - angles[left] > radianAngle) ++left;
39 maxVisible = std::max(maxVisible, right - left + 1);
40 }
41
42 // Return the maximum number of points visible plus any overlapping points
43 return maxVisible + overlappingPoints;
44 }
45};
46
1function visiblePoints(points: number[][], angle: number, location: number[]): number {
2 const angles: number[] = []; // Array to store the angles of the visible points
3 const observerX = location[0], observerY = location[1]; // Observer's location
4 let overlappingPoints = 0; // Count of points that overlap with the observer's location
5
6 for (const point of points) {
7 const pointX = point[0], pointY = point[1];
8 if (pointX === observerX && pointY === observerY) {
9 // Increment count if the point's location overlaps with the observer's location
10 overlappingPoints++;
11 } else {
12 // Calculate the angle from the observer to the point and store it in the array
13 const angleRadians = Math.atan2(pointY - observerY, pointX - observerX);
14 angles.push(angleRadians);
15 }
16 }
17
18 // Sort the angles in ascending order
19 angles.sort((a, b) => a - b);
20
21 const totalAngles = angles.length; // Total number of unique angles
22
23 // Duplicate the angles by adding 2*PI to each to handle the wrap-around effect
24 for (let i = 0; i < totalAngles; i++) {
25 angles.push(angles[i] + 2 * Math.PI);
26 }
27
28 let maxVisible = 0; // Maximum number of points visible within the angle
29 const radianAngle = angle * Math.PI / 180; // Convert angle to radians for comparison
30
31 // Two-pointer technique to find the max number of points that fit within the angle
32 for (let left = 0, right = 0; right < 2 * totalAngles; right++) {
33 // Move the left pointer to ensure all points are within the viewing angle
34 while (angles[right] - angles[left] > radianAngle) {
35 left++;
36 }
37 maxVisible = Math.max(maxVisible, right - left + 1);
38 }
39
40 // Return the sum of the maximum number of visible points and any overlapping points
41 return maxVisible + overlappingPoints;
42}
43
Time and Space Complexity
The given code calculates how many points are visible within a given angle from a particular location.
Time Complexity
-
The first loop iterates over all points counting points that are the same as the
location
and for others, calculates the angle with respect to location. The complexity for this loop isO(n)
wheren
is the number of points since each operation inside the loop takesO(1)
time. -
The
sort
operation on the angles list isO(n log n)
since all non-same points are sorted. -
The list is doubled to handle the circular nature of the problem, which means you're doing another loop of
O(n)
for that addition. However, this does not change the order of complexity given that list concatenation isO(k)
wherek
is the size of the list being added, so it’s stillO(n)
. -
The
max
function involves a generator expression withbisect_right
calls for each of then
elements in the list.bisect_right
has a complexity ofO(log n)
, so this step has a complexity ofO(n log n)
. -
Adding these together, the overall time complexity is dominated by the sorting and binary search steps, both of which are
O(n log n)
. Hence, the final time complexity isO(n log n)
.
Space Complexity
-
v
is a list that stores up ton
angles when none of the points coincide with thelocation
. This makes the space complexityO(n)
. -
After this list
v
is doubled, making the space complexityO(2n)
or simplyO(n)
since constant factors are dropped in Big O notation. -
There are no other data structures that grow with input size.
Hence, the final space complexity is O(n)
.
Learn more about how to find time and space complexity quickly using problem constraints.
You are given an array of intervals where intervals[i] = [start_i, end_i]
represent the start and end of the ith
interval. You need to merge all overlapping intervals and return an array of the non-overlapping intervals that cover all the intervals in the input.
Recommended Readings
Math for Technical Interviews How much math do I need to know for technical interviews The short answer is about high school level math Computer science is often associated with math and some universities even place their computer science department under the math faculty However the reality is that you
Sorting Summary Comparisons We presented quite a few sorting algorithms and it is essential to know the advantages and disadvantages of each one The basic algorithms are easy to visualize and easy to learn for beginner programmers because of their simplicity As such they will suffice if you don't know any advanced
https algomonster s3 us east 2 amazonaws com cover_photos stack svg Sliding Window Maximum Monotonic Stack We have an array and a sliding window defined by a start index and an end index The sliding window moves from left of the array to right There are always k elements in
Got a question? Ask the Monster Assistant anything you don't understand.
Still not clear? Submit the part you don't understand to our editors. Or join our Discord and ask the community.