-
Notifications
You must be signed in to change notification settings - Fork 2
/
Algorithmic skills. PolygonConcavityIndex.swift
123 lines (103 loc) · 5.42 KB
/
Algorithmic skills. PolygonConcavityIndex.swift
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
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
import Foundation
import Glibc
import extratypes // this library contains declarations of types from the task
// Solution @ Sergey Leschev, Belarusian State University
// Algorithmic skills. PolygonConcavityIndex.
// An array A of points in a 2D plane is given. These points represent a polygon: every two consecutive points describe an edge of the polygon, and there is an edge connecting the last point and the first point in the array.
// A set of points in a 2D plane, whose boundary is a straight line, is called a semiplane. More precisely, any set of the form {(x, y) : ax + by ≥ c} is a semiplane. The semiplane contains its boundary.
// A polygon is convex if and only if, no line segment between two points on the boundary ever goes outside the polygon.
// For example, the polygon consisting of vertices whose Cartesian coordinates are consecutively:
// (-1, 3) (3, 1) (0, -1) (-2, 1)
// is convex.
// The convex hull of a finite set of points in a 2D plane is the smallest convex polygon that contains all points in this set. For example, the convex hull of a set consisting of seven points whose Cartesian coordinates are:
// (-1, 3) (1, 2) (3, 1) (1, 1) (0, -1) (-2, 1) (-1, 2)
// is a polygon that has five vertices. When traversed clockwise, its vertices are:
// (-1, 3) (1, 2) (3, 1) (0, -1) (-2, 1)
// If a polygon is concave (that is, it is not convex), it has a vertex which does not lie on its convex hull border. Your assignment is to find such a vertex.
// Assume that the following declarations are given:
// class Point2D {
// public int x;
// public int y;
// }
// Write a function:
// class Solution { public int solution(Point2D[] A); }
// that, given a non-empty array A consisting of N elements describing a polygon, returns −1 if the polygon is convex. Otherwise, the function should return the index of any point that doesn't belong to the convex hull border. Note that consecutive edges of the polygon may be collinear (that is, the polygon might have 180−degrees angles).
// To access the coordinates of the K-th point (where 0 ≤ K < N), use the following syntax:
// A[K].x to access the x-coordinate,
// A[K].y to access the y-coordinate.
// For example, given array A such that:
// A[0].x = -1 A[0].y = 3
// A[1].x = 1 A[1].y = 2
// A[2].x = 3 A[2].y = 1
// A[3].x = 0 A[3].y = -1
// A[4].x = -2 A[4].y = 1
// the function should return −1, as explained in the example above.
// However, given array A such that:
// A[0].x = -1 A[0].y = 3
// A[1].x = 1 A[1].y = 2
// A[2].x = 1 A[2].y = 1
// A[3].x = 3 A[3].y = 1
// A[4].x = 0 A[4].y = -1
// A[5].x = -2 A[5].y = 1
// A[6].x = -1 A[6].y = 2
// the function should return either 2 or 6. These are the indices of the polygon lying strictly in its convex hull (that is, not on the convex hull border).
// Write an efficient algorithm for the following assumptions:
// N is an integer within the range [3..10,000];
// the coordinates of each point in array A are integers within the range [−1,000,000,000..1,000,000,000];
// no two edges of the polygon A intersect, other than meeting at their endpoints;
// array A does not contain duplicate points.
public func solution(_ A: inout [Point2D]) -> Int {
struct Point {
let point: Point2D
let index: Int
}
var points = [Point]()
var lowestPoint = Point(point: A[0], index: 0)
for i in 1..<A.count {
let point = A[i]
if point.y < lowestPoint.point.y {
points.append(lowestPoint)
lowestPoint = Point(point: point, index: i)
} else if point.y == lowestPoint.point.y && point.x < lowestPoint.point.x {
points.append(lowestPoint)
lowestPoint = Point(point: point, index: i)
} else {
points.append(Point(point: point, index: i))
}
}
let radians = { (a: Point, b: Point) -> Double in
return atan2(Double(b.point.y - a.point.y), Double(b.point.x - a.point.x))
}
let distance = { (a: Point, b: Point) -> Double in
return sqrt(pow(Double(b.point.x - a.point.x), 2) + pow(Double(b.point.y - a.point.y), 2))
}
let orientation = { (a: Point, b: Point, c: Point) -> Int in
return (b.point.y - a.point.y) * (c.point.x - b.point.x) - (b.point.x - a.point.x) * (c.point.y - b.point.y)
}
let sortedPoints = points.sorted { (a, b) -> Bool in
let aRadians = radians(lowestPoint, a)
let bRadians = radians(lowestPoint, b)
if aRadians == bRadians {
let aDistance = distance(lowestPoint, a)
let bDistance = distance(lowestPoint, b)
return aDistance < bDistance
} else {
return aRadians < bRadians
}
}
var stack = sortedPoints
stack.insert(lowestPoint, at: 0)
stack.append(lowestPoint)
let maxRadians = radians(lowestPoint, sortedPoints.last!)
while stack.count != 3 {
let a = stack[0]
let b = stack[1]
let c = stack[2]
let orientation = orientation(a, b, c)
if orientation > 0 && maxRadians != radians(lowestPoint, b) {
return b.index
}
stack.removeFirst()
}
return -1
}