Sicily 1-D closet pair

写在前面:题意应该是求closest pair,老师大概觉得这题太naïve了

题目:

Description
Given a set S = {p1, p2, …, pn} of n points in the 1 dimensional space, find the two points of S whose distance is the smallest.

Input
There are multiple cases.
For each case, the first line is the number of points n (2<=n<=100000). The second line contains n real numbers.

Output
For each case, output the smallest distance, the answer is rounded to 6 digits after the decimal point.

Sample Input
4
1 3 6 10

Sample Output
2.000000

Hint
This problem is simple. We require that you use the divide-and-conquer algorithm to solve it. Please do not use any sort functions.

//
题意说明:给你一堆一维坐标上的点,让你求最短的两点间距离

思路:
用分治法且不用sort函数

1.只有一个点,返回无穷
2.只有两个点,返回两点间距离
3.其他情况将问题分割:
分为左右两个区间,分别找出左右区间最短的两点间距离
再找到分立于两区间的最短的两点间距离
三者取最小值

简单看来很简单,其实很多细节要注意
比如,如何分割?
如何找分立于两个区间的点的最小距离?

这里分割直接模仿了快速排序的partition,并且用了简单的版本
其他具体内容见代码

tips:要分清物理上(内存里)一排点,和抽象的数轴上的一排点
做这道题很容易(可能只有我)把内存上相邻的点想成数轴上相邻,而忽略了分割需要调整位置的步骤

#include <cstdio>
#include <cstdlib>
#include <algorithm>
using namespace std;

#define INF 1<<30 
//一个大数,后面会用到

double points[100001];//用于存储输入的点的信息

int partition(int begin, int end) {//对一个给定的区间,找一个主元,把数组分为
    int i = begin - 1;                   //一边小于主元,一边大于,就是快速排序里的分割方法
    for (int j = begin; j < end; j++) {//不过这里直接选最后一个为主元,会引起一些问题
        if (points[j] <= points[end]) {//但是后面会处理
            i++;
            swap(points[i], points[j]);
        }
    }
    swap(points[i + 1], points[end]);
    return i+1;
}

double get_closest(int begin,int end) {
    if (begin-end+1 == 1)//当前区间只有一个点,返回无穷,用一个大数代替
        return INF;
    else if (begin - end + 1 == 2)//两个点,直接返回这两个点距离
        return abs(points[begin]-points[end]);
    else {
        int mid = partition(begin, end);
        //分割区间并得到分割的主元,注意这个主元还不确定是左区间的最后一个点
        int left_end = mid == begin ? mid : mid - 1;
        //由于可能刚好选的主元是原来区间的最小最大值,如果把mid当成左区间的末尾或者
        //右区间的起始,都会有一种情况无法分割造成递归不终止
        //所以刚好是最小值时,就把mid当做左边区间的末尾
        //否则就是右边区间的起始,这样至少会分割一个点出去
        int right_begin = left_end + 1;
        //以上三行就把当前区间分成了左右两部分
        //左:[begin,left_end],右:[right_begin,end]

        double left_max = points[begin], right_min = points[right_begin];
        for (int i = begin + 1; i <= left_end; i++)
            if (points[i] > left_max)
                left_max = points[i];
        for (int i = right_begin + 1; i <= end; i++)
            if (points[i] < right_min)
                right_min = points[i];
        //两个循环找到左区间的最大值和右区间的最小值
        double ans_mid = abs(left_max - right_min);
        //得到两个区间间最近的两个点的距离

        double ans_left = get_closest(begin, left_end);
        double ans_right = get_closest(right_begin, end);
        //递归求得左右区间最短距离

        double ans = ans_left < ans_right ? ans_left : ans_right;
        ans = ans_mid < ans ? ans_mid : ans;
        return ans;
    }
}

int main() {
    int n;
    while (scanf("%d",&n)!=EOF) {
        for (int i = 0; i < n; i++) 
            scanf("%lf", &points[i]);//因为输入还蛮多的,用scanf算了
        printf("%.6lf\n", get_closest(0,n-1));
    }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值