-
问题
最近对问题要求在包含有n个点的集合S中,找出距离最近的两个点。假设 p1(x1,y1),p2(x2,y2),……,pn(xn,yn)是平面的n个点。 -
解析
分治法总体来说分为三个步骤:划分、求解子问题、合并
我对这题的思路是,将集合S分为左右两个子集S1和S2,两个子集分别含有n/2个点,先在每个子集中递归地求其最接近的点对,再求最近的两点分别在两个集合中的点对,最后比较得出对短距离。 -
设计
minDist = infinity
for i = 1 to length(P) - 1
for j = i + 1 to length(P)
let p = P[i], q = P[j]
if dist(p, q) < minDist:
minDist = dist(p, q)
closestPair = (p, q)
return closestPair
-
分析
因此得出复杂度为O(nlogn) -
源码
struct point {
double x;
double y;
point(double x, double y) :x(x), y(y) {}
point() { return; }
};
bool cmp_x(const point & A, const point & B)
{
return A.x < B.x;
}
bool cmp_y(const point & A, const point & B)
{
return A.y < B.y;
}
double distance(const point & A, const point & B)
{
return sqrt(pow(A.x - B.x, 2) + pow(A.y - B.y, 2));
}
tor<point> & points, double dis, int mid)
{
vector<point> left, right;
for (int i = 0; i < points.size(); ++i)
{
if (points[i].x - points[mid].x <= 0 && points[i].x - points[mid].x > -dis)
left.push_back(points[i]);
else if (points[i].x - points[mid].x > 0 && points[i].x - points[mid].x < dis)
right.push_back(points[i]);
}
sort(right.begin(), right.end(), cmp_y);
for (int i = 0, index; i < left.size(); ++i)
{
for (index = 0; index < right.size() && left[i].y < right[index].y; ++index);
for (int j = 0; j < 7 && index + j < right.size(); ++j)
{
if (distance(left[i], right[j + index]) < dis)
dis = distance(left[i], right[j + index]);
}
}
return dis;
}
double closest(vector<point> & points)
{
if (points.size() == 2) return distance(points[0], points[1]);
if (points.size() == 3) return min(distance(points[0], points[1]), min(distance(points[0], points[2]),
distance(points[1], points[2]))); // 三个点
int mid = (points.size() >> 1) - 1;
double d1, d2, d;
vector<point> left(mid + 1), right(points.size() - mid - 1);
copy(points.begin(), points.begin() + mid + 1, left.begin());
copy(points.begin() + mid + 1, points.end(), right.begin());
d1 = closest(left);
d2 = closest(right);
d = min(d1, d2);
return merge(points, d, mid);
}
int main()
{
int count;
printf("点个数:");
scanf("%d", &count);
vector<point> points;
double x, y;
for (int i = 0; i < count; ++i)
{
printf("第%d个点", i);
scanf("%lf%lf", &x, &y);
point p(x, y);
points.push_back(p);
}
sort(points.begin(), points.end(), cmp_x);
printf("最近点对值:%lf", closest(points));
return 0;
}