作业五

1.问题
最近点对问题:n个点在公共空间中,求出所有点对的欧几里得距离最小的点对。
2.解析
(1)分解
对所有的点按照x坐标(或者y)从小到大排序(排序方法时间复杂度O(nlogn) O(nlogn)O(nlogn))。
根据下标进行分割,使得点集分为两个集合。
(2)解决
递归的寻找两个集合中的最近点对。
取两个集合最近点对中的最小值
(3)合并
最近距离不一定存在于两个集合中,可能一个点在集合A,一个点在集合B,而这两点间距离小于dis。
3.设计

double closestPoint(point s[],int low,int high,point rec[]){
    double d1,d2,d3,d;
    int mid,i,j,index;
    double x1,y1,x2,y2;         //记录点对的位置
    point P[high-low+1],temp1[2],temp2[2],temp3[2];         //辅助空间
    if(high-low==1){             //两个点的情况
        rec[0].x=s[low].x;rec[0].y=s[low].y;
        rec[1].x=s[high].x;rec[1].y=s[high].y;
        return Distance(s[low],s[high]);
        }
    if(high-low==2){            //三个点的情况
        d1=Distance(s[low],s[low+1]);
        d2=Distance(s[low+1],s[high]);
        d3=Distance(s[low],s[high]);
        if((d1<d2)&&(d1<d3)){
            rec[0].x=s[low].x;rec[0].y=s[low].y;
            rec[1].x=s[low+1].x;rec[1].y=s[low+1].y;
            return d1;
        }
        else if(d2<d3){
            rec[0].x=s[low+1].x;rec[0].y=s[low+1].y;
            rec[1].x=s[high].x;rec[1].y=s[high].y;
            return d2;
        }
        else {
            rec[0].x=s[low].x;rec[0].y=s[low].y;
            rec[1].x=s[high].x;rec[1].y=s[high].y;
            return d3;
        }
    }
    mid=(low+high)/2;       //其他情况递归
    d1=closestPoint(s,low,mid,rec);
    temp1[0]=rec[0];
    temp1[1]=rec[1];
    d2=closestPoint(s,mid+1,high,rec);
    temp2[0]=rec[0];
    temp2[1]=rec[1];
    if(d1<d2){
        d=d1;
        rec[0]=temp1[0];
        rec[1]=temp1[1];
    }
    else {
        d=d2;
        rec[0]=temp2[0];
        rec[1]=temp2[1];
    }
    index=0;
    for(i=mid;(i>=low)&&((s[mid].x-s[i].x)<d);i--)      //点集合p1
        P[index++]=s[i];
    for(i=mid+1;(i<=high)&&((s[i].x-s[mid].x)<d);i++)      //点集合p2
        P[index++]=s[i];
    sort(P,P+index,cmp);                    //升序排列
    for(i=0;i<index;i++){
        for(j=j+1;j<index;i++){
            if((P[j].y-P[i].y)>=d)
                break;
            else {
                d3=Distance(P[i],P[j]);
                if(d3<d){
                    rec[0].x=P[i].x;rec[0].y=P[i].y;
                    rec[1].x=P[j].x;rec[1].y=P[j].y;
                    d=d3;
                }
            }
        }
    }
    return d;
}

4.分析
O(nlog2,n)
5.源码

`#include<stdio.h>
#include<math.h>
#include<string.h>
#include
#include
using namespace std;
struct point{ //点结构
double x,y;
};
double Distance(point a,point b){
return sqrt((a.x-b.x)(a.x-b.x)+(a.y-b.y)(a.y-b.y));
}
bool cmp(point a,point b){ //按y升排序辅助函数
return a.y<b.y;
}
bool cmp2(point a,point b){ //按x升排序辅助函数
return a.x<b.x;
}
double closestPoint(point s[],int low,int high,point rec[]){
double d1,d2,d3,d;
int mid,i,j,index;
double x1,y1,x2,y2; //记录点对的位置
point P[high-low+1],temp1[2],temp2[2],temp3[2]; //辅助空间
if(high-low1){ //两个点的情况
rec[0].x=s[low].x;rec[0].y=s[low].y;
rec[1].x=s[high].x;rec[1].y=s[high].y;
return Distance(s[low],s[high]);
}
if(high-low
2){ //三个点的情况
d1=Distance(s[low],s[low+1]);
d2=Distance(s[low+1],s[high]);
d3=Distance(s[low],s[high]);
if((d1<d2)&&(d1<d3)){
rec[0].x=s[low].x;rec[0].y=s[low].y;
rec[1].x=s[low+1].x;rec[1].y=s[low+1].y;
return d1;
}
else if(d2<d3){
rec[0].x=s[low+1].x;rec[0].y=s[low+1].y;
rec[1].x=s[high].x;rec[1].y=s[high].y;
return d2;
}
else {
rec[0].x=s[low].x;rec[0].y=s[low].y;
rec[1].x=s[high].x;rec[1].y=s[high].y;
return d3;
}
}
mid=(low+high)/2; //其他情况递归
d1=closestPoint(s,low,mid,rec);
temp1[0]=rec[0];
temp1[1]=rec[1];
d2=closestPoint(s,mid+1,high,rec);
temp2[0]=rec[0];
temp2[1]=rec[1];
if(d1<d2){
d=d1;
rec[0]=temp1[0];
rec[1]=temp1[1];
}
else {
d=d2;
rec[0]=temp2[0];
rec[1]=temp2[1];
}
index=0;
for(i=mid;(i>=low)&&((s[mid].x-s[i].x)<d);i–) //点集合p1
P[index++]=s[i];
for(i=mid+1;(i<=high)&&((s[i].x-s[mid].x)<d);i++) //点集合p2
P[index++]=s[i];
sort(P,P+index,cmp); //升序排列
for(i=0;i<index;i++){
for(j=j+1;j<index;i++){
if((P[j].y-P[i].y)>=d)
break;
else {
d3=Distance(P[i],P[j]);
if(d3<d){
rec[0].x=P[i].x;rec[0].y=P[i].y;
rec[1].x=P[j].x;rec[1].y=P[j].y;
d=d3;
}
}
}
}
return d;
}
int main(){
point p[10]; //设定点的集合
int n;
double minDist;
cout<<“输入点的个数:\n”; //输入点的个数
cin>>n;
cout<<“输入点集:(x,y)\n”;
for(int i=0;i<n;i++)
cin>>p[i].x>>p[i].y;
sort(p,p+n,cmp2); //对输入的点先排序
point index[2];
minDist=closestPoint(p,0,n-1,index);
cout<<“最小距离点对为:(”<<index[0].x<<","<<index[0].y<<"),("<<index[1].x<<","<<index[1].y<<")\n";
cout<<“最小距离为:\n”<<minDist; //输出点对的最小问题
return 0;
}

`

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值