forked from cacophonix/SPOJ
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathBSHEEP.cpp
116 lines (102 loc) · 2.48 KB
/
BSHEEP.cpp
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
/*
USER: zobayer
TASK: BSHEEP
ALGO: convex hull
*/
#include <cstdio>
#include <algorithm>
#include <cmath>
using namespace std;
const int MAX = 100032;
const double EPS = 1e-10;
struct point {
int x, y, id;
};
point P[MAX], C[MAX], P0;
inline int triArea2(const point &a, const point &b, const point &c) {
return (a.x*(b.y-c.y) + b.x*(c.y-a.y) + c.x*(a.y-b.y));
}
inline int sqDist(const point &a, const point &b) {
return ((a.x-b.x)*(a.x-b.x) + (a.y-b.y)*(a.y-b.y));
}
inline double dist(const point &a, const point &b) {
return hypot((double)(a.x-b.x), (double)(a.y-b.y));
}
inline bool comp(const point &a, const point &b) {
int d = triArea2(P0, a, b);
if(d < 0) return false;
if(!d && sqDist(P0, a) > sqDist(P0, b)) return false;
return true;
}
inline bool normal(const point &a, const point &b) {
return ((a.x==b.x) ? a.y < b.y : a.x < b.x);
}
inline bool issame(const point &a, const point &b) {
return (a.x == b.x && a.y == b.y);
}
inline void makeUnique(int &np) {
sort(P, P+np, normal);
np = unique(P, P+np, issame) - P;
}
void convexHull(int &np, int &nc) {
int i, j, pos = 0;
for(i = 1; i < np; i++)
if(P[i].y<P[pos].y || (P[i].y==P[pos].y && P[i].x<P[pos].x))
pos = i;
swap(P[0], P[pos]);
P0 = P[0];
sort(&P[1], &P[np], comp);
for(i = 0; i < 3; i++) C[i] = P[i];
for(i = j = 3; i < np; i++) {
while(triArea2(C[j-2], C[j-1], P[i]) < 0) j--;
C[j++] = P[i];
}
nc = j;
}
void compress(int &nc) {
int i, j, d;
for(i=j=1; i < nc; i++) {
d = triArea2(C[j-1], C[i], C[i+1]);
if(d || (!d && C[j-1].id == C[i+1].id)) C[j++] = C[i];
}
nc = j;
}
int main() {
int i, np, nc, t;
double perim;
//freopen("in.txt", "r", stdin);
//freopen("out.txt", "w", stdout);
scanf("%d", &t);
while(t--) {
scanf("%d", &np);
for(i = 0; i < np; i++) {
scanf("%d%d", &P[i].x, &P[i].y);
P[i].id = i + 1;
}
makeUnique(np);
//for(i = 0; i < np; i++) printf("(%d,%d) ", P[i].x, P[i].y); printf("\n");
if(np==1) {
printf("0.00\n");
printf("%d\n", P[0].id);
}
else if(np==2) {
perim = 2.0 * dist(P[0], P[1]);
printf("%.2lf\n", perim + EPS);
printf("%d %d\n", P[0].id, P[1].id);
}
else {
convexHull(np, nc);
//for(i = 0; i < nc; i++) printf("(%d,%d) ", C[i].x, C[i].y); printf("\n");
C[nc] = C[0];
perim = 0.00;
for(i = 0; i < nc; i++) perim += dist(C[i], C[i+1]);
printf("%.2lf\n", perim + EPS);
compress(nc);
printf("%d", C[0].id);
for(i = 1; i < nc; i++) printf(" %d", C[i].id);
printf("\n");
}
if(t) printf("\n");
}
return 0;
}