주어진 점집합의 Convex hull을 찾는 알고리즘은 보통

// cross(P1-P0, P2-P0);
// = := colinear;
// < := <p0,p1,p2> CW order;
// > := <p0,p1,p2> CCW order;
static
int ccw(const CPoint &P0, const CPoint &P1, const CPoint &P2 ) {
return (P1.x - P0.x)*(P2.y - P0.y) - (P2.x - P0.x)*(P1.y - P0.y);
}
std::vector<CPoint> approximateHull(const std::vector<CPoint>& Q, int bins) {
if (Q.size() < 3) return std::vector<CPoint> ();
if (bins < 0) bins = max(1, Q.size()/10);
int Lmin = 0, Lmax = 0, Rmin = 0, Rmax = 0;
int left = Q[0].x, right = left;
for (int i = 1; i < Q.size(); i++) {
if (Q[i].x <= left) {
if (Q[i].x < left) {
Lmin = Lmax = i;
left = Q[i].x;
}
else
if (Q[i].y < Q[Lmin].y) Lmin = i;
else if (Q[i].y > Q[Lmax].y) Lmax = i;
}
if (Q[i].x >= right) {
if (Q[i].x > right) {
Rmin = Rmax = i;
right = Q[i].x;
}
else
if (Q[i].y < Q[Rmin].y) Rmin = i;
else if (Q[i].y > Q[Rmax].y) Rmax = i;
}
}
if (left == right) return std::vector<CPoint> (); //vertical line;
//
std::vector<int> slotYhigh(bins + 2, -1);
std::vector<int> slotYlow(bins + 2, -1);
slotYlow.front() = Lmin;
slotYhigh.front() = Lmax;
slotYlow.back() = Rmin;
slotYhigh.back() = Rmax;
//
const CPoint &A = Q[Lmin];
const CPoint &B = Q[Rmin];
const CPoint &C = Q[Lmax];
const CPoint &D = Q[Rmax];
for (int i = 0; i < Q.size(); i++) {
if (Q[i].x == right || Q[i].x == left) continue;
if (ccw(A, B, Q[i]) < 0) { // below line(A,B);
int b = 1 + (bins * (Q[i].x - left)) / (right - left);
if (slotYlow[b]==-1) slotYlow[b] = i;
else if (Q[i].y < Q[slotYlow[b]].y) slotYlow[b] = i;
continue ;
}
if (ccw(C, D, Q[i]) > 0) {// above line(C,D);
int b = 1 + (bins * (Q[i].x - left)) / (right - left);
if (slotYhigh[b]==-1) slotYhigh[b] = i;
else if (Q[i].y > Q[slotYhigh[b]].y) slotYhigh[b] = i;
continue;
}
}
std::vector<CPoint> hull(Q.size());
int s = -1;
for (int i = 0; i <= bins + 1; i++) {
if (slotYlow[i]==-1) continue;
while (s > 0)
if (ccw(hull[s-1], hull[s], Q[slotYlow[i]]) > 0) break;
else s--;
hull[++s] = Q[slotYlow[i]];
};
if (Rmax != Rmin) hull[++s] = Q[Rmax];
//
int s0 = s ;
for (int i = bins; i >= 0; i--) {
if (slotYhigh[i]==-1) continue;
while (s > s0)
if (ccw(hull[s-1], hull[s], Q[slotYhigh[i]]) > 0) break;
else s--;
hull[++s] = Q[slotYhigh[i]];
}
if (hull[s] == hull[0]) s--;
hull.resize(s+1);
return hull;
}
'Computational Geometry' 카테고리의 다른 글
Alpha Shape (1) | 2024.07.14 |
---|---|
Smoothing Spline (0) | 2024.06.29 |
Catmull-Rom Spline (2) (0) | 2024.06.21 |
Minimum Volume Box (0) | 2024.06.16 |
Natural Cubic Spline: revisited (0) | 2024.06.14 |