점집합을 x-좌표의 값(같으면, y좌표의 값) 순으로 정렬한다. 시작점은 hull의 꼭짓점이 된다.
x-좌표 순서대로 현재 점이 hull의 꼭짓점인가를 검사한다: hull의 꼭짓점이 되기 위해서는 직전까지 구성된 hull의 에지 왼편에 있어야 한다. 오른편에 있으면 직전까지 구성된 hull에서 현재 점이 왼편에 놓일 때까지 꼭짓점을 제거한다.
남는 점들을 1번 과정의 역순으로 정렬한다. 최대 x-좌표, 최소 y-좌표점이 다시 윗 hull의 시작점이 된다.
x-좌표가 작아지는 순서대로 2번 과정을 수행한다.
staticintccw(const CPoint& A, const CPoint& B, const CPoint& C){
// cross (C-A, B-A) >= 0 이면 AB의 오른쪽에 C가 놓임;int cax = C.x - A.x;
int cay = C.y - A.y;
int bax = B.x - A.x;
int bay = B.y - A.y;
return (cax * bay - cay * bax) >= 0;
}
// lower-hull은 x-증가: y-감소로 정렬;staticintcmpl(constvoid *a, constvoid *b){
int v = ((CPoint *)a)->x - ((CPoint *)b)->x;
if (v > 0) return1;
if (v < 0) return-1;
v = ((CPoint *)b)->y - ((CPoint *)a)->y;
if (v > 0) return1;
if (v < 0) return-1;
return0;
}
// upper-hull은 x-감소: y-증가로 정렬;staticintcmph(constvoid *a, constvoid *b){
returncmpl(b, a);
}
staticintmakeChain(CPoint *V, int n, int (*cmp)(constvoid*, constvoid*)){
qsort(V, n, sizeof(CPoint), cmp);
// 점 i (>j))가 현재까지 생성된 hull(s까지)의 에지보다도 더 오른쪽에 있으면, // 왼쪽에 위치할 때까지 기존 hull을 줄이고. (i)를 hull에 추가.int s = 1;
for (int i = 2; i < n; i++) {
int j = s;
while(j >= 1 && ccw(V[j-1], V[j], V[i])) j--;
// ccw의 =때문에 일직선상의 마지막점만 들어감;
s = j + 1; // j = last index of hull;// (j+1)에 새 hull-point을 추가:
CPoint t = V[i]; V[i] = V[s]; V[s] = t;
}
return s;
}
std::vector<CPoint> chainHull2(const std::vector<CPoint>& P){
if (P.size() < 3) return std::vector<CPoint> (); //null_vector;// copy to working array;std::vector<CPoint> chull(P.size()+1);
for (int i = P.size(); i-->0;) chull[i] = P[i];
/* make lower hull;*/int u = makeChain(&chull[0], P.size(), cmpl);
/* make upper hull;*/
chull.back()= chull.front();
u += makeChain(&chull[u], P.size() - u + 1, cmph);
chull.resize(u);
return chull;
}
Input = a set S of n points
Assume that there are at least 2 points in the input set S of points
QuickHull (S) {
// Find convex hull from the set S of n points
Convex Hull := {}
가장 왼쪽(A)과 오른쪽(B)에 있는 두 점을 찾아 convec hull에 추가;
선분 AB을 기준으로 나머지 (n-2)개 점들을 두 그룹 S1과 S2로 나눔;
S1 = 선분 AB의 오른쪽에 놓인 점;
S2 = 선분 BA의 오른쪽에 놓인 점;
FindHull (S1, A, B)
FindHull (S2, B, A)
};
FindHull (Sk, P, Q) {
If Sk has no point,
then return.
선분 PQ에서 가장 멀리 떨어진 점(C)을 찾아 convex hull에서 P와 Q의 사이에 추가;
세점 P, C, Q는 나머지 Sk의 나머지 점들을 세 그룹S0, S1, and S2
S0 = 삼각형 PCQ에 포함되는 점;
S1 = 선분 PC의 오른쪽에 놓인 점;
S2 = 선분 CQ의 오른쪽에 놓인 점;
FindHull(S1, P, C)
FindHull(S2, C, Q)
}
Output = convex hull
staticvoidfindHull(CPoint *S, int n, CPoint A, CPoint B, std::vector<CPoint>& hull){
if (n < 1) return ;
//Find furtherst point double maxdist = 0;
for (int i = 0; i < n; i++) {
double dist = dist2ToLine(S[i], A, B);
if (dist > maxdist) {
SWAP_POINT(S[0], S[i]);
maxdist = dist;
}
}
if (maxdist == 0.) return ; //collinear case;
hull.push_back(S[0]);
int j = 1;
for (int i = 1; i < n; i++) {
if (leftSide(S[i], A, S[0])) {
SWAP_POINT(S[i], S[j]);
j++;
};
};
int k = j;
for (int i = j; i < n; i++) {
if (leftSide(S[i], S[0], B)) {
SWAP_POINT(S[i], S[k]);
k++;
}
};
//1,...,j-1;findHull(&S[1], j-1, A, S[0], hull);
//j,...,k-1;findHull(&S[j], k-j, S[0], B, hull);
};
voidfindMaxMin(std::vector<CPoint>& pts){
int minx = pts[0].x, maxx = minx;
int minid = 0, maxid = 0;
for (int i = pts.size(); i-->1;) {
if (pts[i].x > maxx) maxid = i, maxx = pts[i].x;
if (pts[i].x < minx) minid = i, minx = pts[i].x;
}
SWAP_POINT(pts[0], pts[minid]); if (maxid == 0) maxid = minid;
SWAP_POINT(pts[1], pts[maxid]);
};
void hullToPolyline(std::vector<CPoint>& hull);
std::vector<CPoint> quickHull(const std::vector<CPoint>& pts){
if (pts.size() < 3) std::vector<CPoint> ();//null_vector;//Find left and right most points, say A & B, and add A & B to convex hull ;findMaxMin(pts);
std::vector<CPoint> hull;
hull.push_back(pts[0]);
hull.push_back(pts[1]);
int j = 2;
for (int i = 2; i < pts.size(); i++) {
if (leftSide(pts[i], pts[0], pts[1])) {
SWAP_POINT(pts[i], pts[j]);
j++;
}
}
//2,3,...,j-1;findHull(&pts[2], j-2, pts[0], pts[1], hull);
//j,j+1,...,pts.size()-1;if (j < pts.size()) // in order to avoid dereferencing &pts[pts.size()];findHull(&pts[j], pts.size()-j, pts[1], pts[0], hull);
hullToPolyline(hull);
return hull;
};
출력 hull은 단순히 convex hull을 구성하는 정렬이 안된 점들만 주므로, hull의 에지를 구하고 싶으면 추가적인 수정이 필요함.
staticintcmph(constvoid *a, constvoid *b){
CPoint *A = (CPoint *)a , *B = (CPoint *)b ;
int v = (A->x - B->x) ;
if (v > 0 ) return1;
if (v < 0 ) return-1;
v = B->y - A->y ;
if (v > 0 ) return1;
if (v < 0 ) return-1;
return0;
};
staticintcmpl(constvoid * a, constvoid *b){
returncmph(b, a);
};
voidhullToPolyline(std::vector<CPoint>& hull){
CPoint A = hull[0];
CPoint B = hull[1];
// 가장 멀리 떨어진 두 점(hull[0], hull[1])을 연결하는 직선을 기준으로 프로젝션을 구하여서 순차적으로 int k = 2;
for (int i = 2; i < hull.size(); i++) {
if (leftSide(hull[i], A, B)) {
SWAP_POINT(hull[i], hull[k]); k++;
};
};
// k-1; last index of hull left side of line(A,B);// upper part reordering: qsort(&hull[0], k, sizeof(CPoint), cmph);
//lower part reordering;if (k < hull.size())
qsort(&hull[k], hull.size()-k, sizeof(CPoint), cmpl);
}
};
또한 입력점들의 순서를 그대로 유지하고 싶으면, double pointer를 이용하거나 복사복은 이용하여야 한다.