-
Notifications
You must be signed in to change notification settings - Fork 278
/
Copy pathErectTheFence.java
74 lines (65 loc) · 2.01 KB
/
ErectTheFence.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
class Solution {
private class Point{
int x;
int y;
Point(int x, int y){
this.x = x;
this.y = y;
}
}
public int[][] outerTrees(int[][] trees) {
List<Point> points = new ArrayList<>();
for(int[] point: trees){
points.add(new Point(point[0], point[1]));
}
List<Point> res = new ArrayList<>();
if (points.size() == 1) {
return trees;
}
int n = points.size();
Collections.sort(points,(a,b) -> a.y == b.y ? a.x-b.x:a.y-b.y);
HashSet<ArrayList<Integer>> dup = new HashSet();
Stack<Point> hull = new Stack<>();
hull.push(points.get(0));
hull.push(points.get(1));
// Lower hull
for (int i = 2; i < n; i++) {
Point top = hull.pop();
while (!hull.isEmpty() && ccw(hull.peek(), top, points.get(i)) < 0) {
top = hull.pop();
}
hull.push(top);
hull.push(points.get(i));
}
// Upper hull
for(int i = n-2;i>=0;i--) {
Point top = hull.pop();
while (!hull.isEmpty() && ccw(hull.peek(), top, points.get(i))< 0) {
top = hull.pop();
}
hull.push(top);
hull.push(points.get(i));
}
for(Point p: hull) {
ArrayList<Integer> tmp = new ArrayList<>();
tmp.add(p.x);tmp.add(p.y);
if(dup.contains(tmp)) continue;
dup.add(tmp);
res.add(p);
}
int[][] ans = new int[res.size()][2];
int i=0;
for(Point p: res){
ans[i][0] = p.x;
ans[i][1]=p.y;
i++;
}
return ans;
}
public int ccw(Point a, Point b, Point c) {
double crossproduct = (b.x-a.x)*(c.y-a.y) - (b.y-a.y)*(c.x-a.x);
if (crossproduct < 0) return -1;
else if (crossproduct > 0) return +1;
else return 0; // linear in natural
}
}