最近点对问题(算法与数据结构设计)

课题内容和要求

最近点对问题,在二维平面上输入n个点列P。其中任一点pi=(xi,yi),编写程序求出最近的两个点。使用穷举法实现,算法复杂度O(n2);优化算法,以O(nlog2n)实现这一问题

数据结构说明

Point_X结构体,用于表示点集,n用来存放点集中点的数量,dis_min用来存放求得的最近点对的距离,p数组用于存放所有点的x轴和y轴坐标。

Point结构体,用于存放点的x轴和y轴坐标。

Point_Y结构体,辅助数据结构,用于求分治法中计算跨越左子部分和右子部分存在的解,其中flag用于标记属于左子部分或右子部分,0表示左子部分,1表示右子部分。

算法设计

穷举法求最近点对,利用双重循环嵌套求每个点与其余点的距离,每当有最小的距离出现,记录两个点以及它们的最小距离。

分治法求最近点对,在主程序中对点集X按x轴由小到大排序,以及对点集X的复制点集Y按y轴由小到大排序,在递归函数中不断递归求中点X[mid]并分划为两部分,直至点被分划到少于或等于3个点,直接求它们中的最近点对以及最短距离min。递的部分完成后开始归,从左右子部分当中取最小的min,并在点集Y中直接取属于(X[mid]-min,X[mid]+min)的点存放到点集Y_中,并把属于(X[mid]-min,X[mid])的左子部分做标记flag=0,把属于(X[mid+1],X[mid]+min)的右子部分做标记flag=1。接着在点集Y_中找属于左子部分的点,每次找到一个属于左子部分的点,在此基础向上找(因为按y轴排序)四个属于右子部分的点,向下找四个属于右子部分的点,一共8个点,计算左子部分找到点与其对应向上向下的8个点的距离求最短距离。

详细设计

程序包含文件main.c、point.h、point.c

main.c

#include <stdlib.h>
#include <stdio.h>
#include "point.h"
#include <math.h>
int main()
{
    Point_X *X = (Point_X*)malloc(sizeof(Point_X));
    #if 1   // 免输入直接出结果,已给出点集合
    double p[10][2] = {{0,5},{-2,2},{0,1},{2,1},{-5,0},{-0.5,0},{0,0},{0.5,0},{5,0},{0,-5}};
    if(!Init(X,10))
    {
        printf("点集初始化失败\n");
        return 0;
    }
    Point *Y = (Point*)malloc((X->n)*sizeof(Point));
    if(!Y)
    {
        printf("申请空间失败——辅助数据Y\n");
    }
    for(int i = 0;i<(X->n);i++)
    {
        X->p[i].x = p[i][0];
        X->p[i].y = p[i][1];
        Y[i].x = p[i][0];
        Y[i].y = p[i][1];
    }
    #elif 0   // 需要用户输入
    int n = 0;  // 存放点集的数量,用来初始化点集
    printf("请输入点集中点的数量:\n");
    scanf("%d",&n);
    if(!Init(X,n))
    {
        printf("点集初始化失败\n");
        return 0;
    }
    Point *Y = (Point*)malloc((X->n)*sizeof(Point));
    if(!Y)
    {
        printf("申请空间失败——辅助数据Y\n");
    }
    for(int i = 0;i<(X->n);i++)
    {
        printf("请输入第%d个点的x轴:",i+1);
        scanf("%lf",&(X->p[i].x));
        printf("请输入第%d个点的y轴:",i+1);
        scanf("%lf",&(X->p[i].y));
        Y[i].x = X->p[i].x;
        Y[i].y = X->p[i].y;
    }
    getchar();  // 将上一个输入的回车接收
    #endif
    // 点对1,用于记录穷举法得出的最近点对
    Point pair_pp1;
    Point pair_pp2;
    // 点对2,用于记录分治法求得出得最近点对
    Point pair_p1;
    Point pair_p2;
    double min = INFINITY;
    for(int i = 0;i<X->n-1;i++)    // 穷举法求最近点对
    {
        for(int j=i+1 ;j<X->n;j++)
        {
            double dis = distance(X->p[i],X->p[j]);
            if(dis<min) // 当dis小于min值更新最近点对的点以及两点之间的距离
            {
                pair_pp1 = X->p[i];
                pair_pp2 = X->p[j];
                min = dis;
            }
        }
    }
    printf("以下为穷举法求得的最近点对:\n");
    printf("点集中最近的两个点,点1:(%.2lf,%.2lf),点2:(%.2lf,%.2lf),两点距离:%.2lf\n",pair_pp1.x,pair_pp1.y,pair_pp2.x,pair_pp2.y,min);
    printf("---------------------------\n");
    QuickSort_X(X->p,0,X->n-1); // 调用针对x轴排序的快速排序算法对点集X排序
    QuickSort_Y(Y,0,X->n-1);    // 调用针对y轴排序的快速排序算法对点集Y排序
    close_point(X,Y,0,X->n-1,&pair_p1,&pair_p2);    // 调用分治法求最近点对
    printf("以下为分治法求得的最近点对:\n");
    printf("点集中最近的两个点,点1:(%.2lf,%.2lf),点2:(%.2lf,%.2lf),两点距离:%.2lf\n",pair_p1.x,pair_p1.y,pair_p2.x,pair_p2.y,X->dis_min);
    printf("输入回车结束程序。\n");    
    getchar();  // 输入回车结束程序
    free(Y);
    free(X->p);
    free(X);
}

point.h

#ifndef __POINT_H__
#define __POINT_H__
typedef struct point
{
    double x;
    double y;
}Point;
typedef struct point_x
{
    Point *p;   // 指向点数组Point的头指针
    double dis_min; // 记录点集中两点最小的距离
    int n;  // 点的数量
}Point_X;
typedef struct point_y  // 辅助数据结构
{
    Point p;
    int flag;   // 表示位,0表示属于左半边,1表示属于右半边
}Point_Y;
int Init(Point_X *X, int n);    // 初始化点集
void Swap(Point *p,int i, int j);   // 交换函数
int Partition_X(Point *p,int left,int right);   // 分化函数,针对x轴
void QuickSort_X(Point *p, int left, int right);    // 快速排序,针对x轴进行排序
int Partition_Y(Point *p,int left,int right);   // 分划函数,针对y轴分划
void QuickSort_Y(Point *p, int left, int right);    // 快速排序,针对y轴排序
double distance(Point p1, Point p2);    // 求点的距离
void close_point(Point_X *X, Point *Y, int left, int right, Point *pair_p1,Point *pair_p2); // 分治法求最近点对
#endif

point.c

#include "point.h"
#include <math.h>
#include <stdlib.h>
#include <stdio.h>
/**
 * @brief 初始化点集
 * @param X 指向点集的指针
 * @param n 点的数量
 * @return 0,失败;1,成功
*/
int Init(Point_X *X, int n)
{
    X->p = (Point*)malloc(n*sizeof(Point));
    if(!X->p)
    {
        return 0;
    }
    X->dis_min = INFINITY;
    X->n = n;
    return 1;
}
/**
 * @brief 交换函数
 * @param p 点数组
 * @param i 数组下标i
 * @param j 数组下标j
 * @return void
*/
void Swap(Point *p,int i, int j)
{
    Point tmp;
    if(i==j)
    {
        return;
    }
    tmp = p[i];
    p[i] = p[j];
    p[j] = tmp;
}
/**
 * @brief 分化函数,针对x轴
 * @param p 点数组
 * @param left 左边界
 * @param right 右边界
 * @return j,分划元素下标
*/
int Partition_X(Point *p,int left,int right)
{   
    int i = left;
    int j = right+1;
    do
    {
        do
        {
            i++;
        }while(p[i].x<p[left].x&&i<=right);
        do
        {
            j--;
        } while(p[j].x>p[left].x);
        if(i<j)
        {
            Swap(p,i,j);
        }             
    }while(i<j);
    Swap(p, left, j);
    return j;
}
/**
 * @brief 快速排序,针对x轴进行排序
 * @param p 点数组
 * @param left 左边界
 * @param right 右边界
 * @return void
*/
void QuickSort_X(Point *p, int left, int right)
{
    if(left<right)
    {
        int j = Partition_X(p, left, right);
        QuickSort_X(p, left, j-1);
        QuickSort_X(p, j+1,right);
    }
}
/**
 * @brief 分划函数,针对y轴分划
 * @param p 点数组
 * @param left 左边界
 * @param right 右边界
 * @return j,分划元素下标
*/
int Partition_Y(Point *p,int left,int right)
{   
    int i = left;
    int j = right+1;
    do
    {
        do
        {
            i++;
        }while(p[i].y<p[left].y&&i<=right);
        do
        {
            j--;
        } while(p[j].y>p[left].y);
        if(i<j)
        {
            Swap(p,i,j);
        }             
    }while(i<j);
    Swap(p,left,j);
    return j;
}
/**
 * @brief 快速排序,针对y轴排序
 * @param p 点数组
 * @param left 左边界
 * @param right 右边界
 * @return void
*/
void QuickSort_Y(Point *p, int left, int right)
{
    if(left<right)
    {
        int j = Partition_Y(p, left, right);
        QuickSort_Y(p, left, j-1);
        QuickSort_Y(p, j+1,right);
    }
}
/**
 * @brief 求两点的距离
 * @param p1 点1
 * @param p2 点2
 * @return 返回两点的距离
*/
double distance(Point p1, Point p2)
{
    return sqrt(pow((p1.x-p2.x),2)+pow((p1.y-p2.y),2));
}
/**
 * @brief 分治法求最近点对
 * @param X 对于x轴已经排序好的点集
 * @param Y 对于y轴已经排序好的点集
 * @param left X点集左边界
 * @param right X点集右边界
 * @param pair_1 最近点对的点1
 * @param pair_2 最近点对的点2
 * @return void
*/
void close_point(Point_X *X, Point *Y, int left, int right, Point *pair_p1,Point *pair_p2)
{
    if(right-left<=2)   // 少于或等于三个点直接计算
    {
        if(left==right) // 只有一个点,直接返回
        {
            return;
        }
        for(int i = left;i<right;i++)
        {
            for(int j = i+1;j<right+1;j++)
            {
                double dis = distance(X->p[i],X->p[j]);
                if(X->dis_min>dis)
                {
                    X->dis_min = dis;
                    pair_p1->x = X->p[i].x;
                    pair_p1->y = X->p[i].y;
                    pair_p2->x = X->p[j].x;
                    pair_p2->y = X->p[j].y;
                }
            }
        }
        return;
    }
    int mid = (right+left)/2;   // 分划点
    close_point(X,Y,left,mid,pair_p1,pair_p2);  // 对左边集合进行计算
    close_point(X,Y,mid+1,right,pair_p1,pair_p2);   // 对右边进行计算
    int num_point = 1;  // 用于计算在范围(X[mid]-min,X[mid]+min)内的点的数量
    int mid_ = mid-1;
    while(X->p[mid_].x>(X->p[mid].x-X->dis_min)&&mid_>=0) // 计算在(X[mid]-min,X[mid])范围的点
    {
        num_point++;
        mid_--;
    }
    mid_ = mid+1;
    while(X->p[mid_].x<X->p[mid].x+X->dis_min&&mid_<=X->n)  // 计算在(X[mid],X[mid]+min)范围的点
    {
        num_point++;
        mid_++;
    }

    // 在Y点集中提取属于(X[mid]-min,X[mid]+mid)的点,并创建Y_集合存放
    Point_Y *Y_ = (Point_Y*)malloc(num_point*sizeof(Point_Y));
    if(!Y_)
    {
        printf("申请空间失败——Y_\n");
    }
    int j = 0;  // Y_点集下标
    for(int i = 0;i<X->n;i++)
    {
        if(Y[i].x>((X->p[mid].x)-(X->dis_min))&&Y[i].x<=X->p[mid].x)  // 属于(X[mid]-min,X[mid])存入Y_集合并做标记
        {
            Y_[j].p = Y[i];
            Y_[j].flag = 0;
            j++;
        }
        if(Y[i].x>X->p[mid].x&&Y[i].x<X->p[mid].x+X->dis_min)   // 属于(X[mid],X[mid]+min)存入Y_集合并做标记
        {
            Y_[j].p = Y[i];
            Y_[j].flag = 1;
            j++;
        }
    }
    for(int i = 0;i<num_point;i++) // 计算(X[mid]-min,X[mid]+min)范围内的最小距离
    {
        if(Y_[i].flag==0)   // 取点集合Y_属于(X[mid]-min,X[mid])的点,并计算每个点与属于(X[mid],X[mid]+min)的最多8个点进行计算
        {
            int four = 0;   // 用于计数
            for(int j = i-1;j>=0;j--)   // 向前找最多4个点进行计算
            {
                if(Y_[j].flag==0)
                {
                    continue;
                }
                four++;
                double dis = distance(Y_[i].p,Y_[j].p);
                if(X->dis_min>dis)
                {
                    X->dis_min = dis;
                    pair_p1->x = X->p[i].x;
                    pair_p1->y = X->p[i].y;
                    pair_p2->x = X->p[j].x;
                    pair_p2->y = X->p[j].y;
                }
                if(four==4) // 最多找4个点,退出该循环
                {
                    four=0;
                    break;
                }
            }
            four=0;
            for(int j = i+1;j<num_point;j++)    // 向后找最多4个点进行计算
            {
                if(Y_[j].flag==0)
                {
                    continue;
                }
                four++;
                double dis = distance(Y_[i].p,Y_[j].p);
                if(X->dis_min>dis)
                {
                    X->dis_min = dis;
                    pair_p1->x = X->p[i].x;
                    pair_p1->y = X->p[i].y;
                    pair_p2->x = X->p[j].x;
                    pair_p2->y = X->p[j].y;
                }
                if(four==4) // 最多找4个点,退出该循环
                {
                    four = 0;
                    break;
                }
            }
four=0;

        }
    }
    free(Y_);
    return;
}

测试数据及其结果分析

例子1,程序中自带的例子:
(0,5)(-2,2)(0,1)(2,1)(-5,0)(-0.5,0)(0,0)(0.5,0)(5,0)(0,-5)

运行结果:
在这里插入图片描述

例子2,手动输入(程序中默认启用例子1,需要在main.c文件中将条件编译中if后的1改为0,elif后的0改为1):
(0,0)(1,1)(0.25,0)(1.1,1)(-2,1)(2,2)(3,3)(-1,-1)

运行结果:
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/784790.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

阶段三:项目开发---民航功能模块实现:任务24:航空实时监控

任务描述 内 容&#xff1a;地图展示、飞机飞行轨迹、扇区控制。航空实时监控&#xff0c;是飞机每秒发送坐标&#xff0c;经过终端转换实时发送给塔台&#xff0c;为了飞机位置的精准度&#xff0c;传输位置的密度很大&#xff0c;在地图位置显示不明显。本次为了案例展示效…

AI系统的PyTorch:TextGrad框架基于文本梯度实现大语言模型AI系统自优化!

AI系统的PyTorch&#xff1a;TextGrad框架基于文本梯度实现大语言模型AI系统自优化&#xff01; 原创 旺知识 旺知识 2024年07月07日 16:21 广东 人工智能&#xff08;AI&#xff09;正在经历一场范式转变&#xff0c;这一转变是由系统协调多个大型语言模型&#xff08;LLMs&…

51 单片机[7]:计时器

一、定时器 1. 定时器介绍 51单片机的定时器属于单片机的内部资源&#xff0c;其电路的连接和运转均在单片机内部完成。 定时器作用&#xff1a; &#xff08;1&#xff09;用于计时系统&#xff0c;可实现软件计时&#xff0c;或者使程序每隔一固定时间完成一项操作 &#…

【零基础】学JS之APIS(基于黑马)

喝下这碗鸡汤 披盔戴甲,一路勇往直前! 1. 什么是事件 事件是在编程时系统内发生的动作或者发生的事情 比如用户在网页上单击一个按钮 2. 什么是事件监听? 就是让程序检测是否有事件产生&#xff0c;一旦有事件触发&#xff0c;就立即调用一个函数做出响应&#xff0c;也称为 注…

【人工智能】—基于成都市各区(市)县租房价格预测建模研究

引言 随着城市化进程的加速&#xff0c;人口流动日益频繁&#xff0c;租房市场作为城市生活的重要组成部分&#xff0c;其价格波动对居民生活质量和城市经济发展具有显著影响。成都市&#xff0c;作为中国西部地区的经济、文化、交通和科技中心&#xff0c;近年来吸引了大量人…

5.Python学习:面向对象

1.面向对象和面向过程的区别 以下五子棋为例&#xff1a; 2.类和实例 &#xff08;1&#xff09;类是抽象的模板&#xff0c;实例是根据模板创建出来的具体的对象 &#xff08;2&#xff09;比如人类就是一个类&#xff0c;刘亦菲就是人类的一个实例 2.1 新建类和类的实例…

王老师 linux c++ 通信架构 笔记(三)安装 xftp、

&#xff08;11&#xff09;调整 xshell 终端的字体大小&#xff0c;默认字体大小是 9 &#xff1a; &#xff08;12&#xff09; 共享文件夹 hgfs 的含义&#xff1a; &#xff08;13&#xff09;安装 xftp &#xff0c; 傻瓜式安装&#xff0c;出了修改下默认安装位置。 操作…

上位机图像处理和嵌入式模块部署(mcu项目2:串口日志记录器)

【 声明&#xff1a;版权所有&#xff0c;欢迎转载&#xff0c;请勿用于商业用途。 联系信箱&#xff1a;feixiaoxing 163.com】 淘宝上面有一个商品蛮好玩的&#xff0c;那就是日志记录器。说是记录器&#xff0c;其实就是一个模块&#xff0c;这个模块的输入是一个ttl串口&am…

18.动态规划之斐波那契数列模型1

1.第N个斐波那契数 1137. 第 N 个泰波那契数 - 力扣&#xff08;LeetCode&#xff09; 做题流程 1. 状态表示&#xff1a; 这道题可以【根据题目的要求】直接定义出状态表示&#xff1a; dp[i] 表示&#xff1a;第 i 个泰波那契数的值。 2. 状态转移方程&#xff1a; …

Social to Sales全链路,数说故事专享会开启出海新视角

————瞎出海&#xff0c;必出局 TikTok&#xff0c;这个充满活力的短视频平台&#xff0c;已经成为全球范围内不可忽视的电商巨头。就在6月8日&#xff0c;TikTok美区带货直播诞生了首个“百万大场”。在此之前&#xff0c;百万GMV被视为一道难以逾越的高墙。以TikTok为首的…

Zabbix分布式监控

目录 分布式监控架构 实现分布式监控的步骤 优点和应用场景 安装Zabbix_Proxy Server端Web页面配置 测试 Zabbix 的分布式监控架构允许在大规模和地理上分散的环境中进行高效的监控。通过分布式监控&#xff0c;Zabbix 可以扩展其监控能力&#xff0c;支持大量主机和设备…

Android - 云游戏本地悬浮输入框实现

一、简述 云游戏输入法分两种情况,以云化原神为例,分为 云端输入法 和 本地输入法,运行效果如下: 云端输入法本地输入法云端输入法 就是运行在云端设备上的输入法,对于不同客户端来说(Android、iPhone),运行效果一致。 本地输入法 则是运行在用户侧设备上的输入法,对…

WordPress开发进群V2主题源码,多种引流方法,引私域二次变现

WordPress开发进群V2主题源码&#xff0c;多种引流方法&#xff0c;引私域二次变现 全新前端UI界面&#xff0c;多种前端交互特效让页面不再单调&#xff0c;进群页面群成员数&#xff0c;群成员头像名称&#xff0c;每次刷新页面随机更新不重复&#xff0c;最下面评论和点赞也…

C语言编程3:运算符,运算符的基本用法

C语言3&#x1f525;&#xff1a;运算符&#xff0c;运算符的基本用法 一、运算符&#x1f33f; &#x1f387;1.1 定义 运算符是指进行运算的动作&#xff0c;比如加法运算符"“&#xff0c;减法运算符”-" 算子是指参与运算的值&#xff0c;这个值可能是常数&a…

4.动态SQL(if,choose,where,set,trim,foreach遍历)的使用+$和#的区别

文章目录 动态sql一、动态sql1.if条件判断2、choose、when、otherwise3、where标签4、set标签5、trim标签1)替代where标签效果2) 生成set标签效果 6、foreach迭代遍历1)属性 7.SQL标签-提取重用的SQL代码片段8、bind标签9.MyBatis中${}和#{}的区别: 动态sql 一、动态sql 常见…

React -- useState状态更新异步特性——导致获取值为旧值的问题

useState状态异步更新 问题导致的原因解决办法进一步分析后续遇到的新问题 问题 const [isSelecting, setIsSelecting] useState(false);useEffect(() > {const handleKeyDown (event) > {if (event.key Escape) {if(isSelectingRef){//.......setIsSelecting(!isSele…

js使用proxy代理监听控制事件

本文为proxy代理的实例应用&#xff0c;有关代理的内容可以参考&#xff1a; js语法---理解反射Reflect对象和代理Proxy对象 监听事件 要监听dom元素的事件&#xff0c;我们会采用回调触发的方式来执行操作&#xff0c; 而触发事件的过程很明显是一个异步操作&#xff0c;异…

Oracle中EXIT Statement用于终止循环语句的关键字

Oracle的EXIT Statement是PL/SQL编程语言中用于终止循环语句的关键字。它有两种主要形式&#xff1a;无条件EXIT和条件EXIT WHEN。以下是对Oracle EXIT Statement的详细解释&#xff1a; 1. 无条件EXIT 语法&#xff1a;EXIT; 作用&#xff1a;无条件地终止当前循环。当程序执…

【咨询】企业数字档案馆(室)建设方案-模版范例

导读&#xff1a;本模版来源某国有大型医药行业集团企业数字档案馆&#xff08;室&#xff09;建设方案&#xff08;一期300W、二期250W&#xff09;&#xff0c;本人作为方案的主要参与者&#xff0c;总结其中要点给大家参考。 目录 1、一级提纲总览 2、项目概述 3、总体规…

办公必备——ONLYOFFICE8.1版本桌面编辑器

一、介绍ONLYOFFICE ONLYOFFICE是一款免费的开源办公软件&#xff0c;它可以让你创建、编辑和分享文档、表格和演示文稿。就像微软的Office一样&#xff0c;但它是完全免费的&#xff0c;而且可以在多种设备上使用&#xff0c;包括电脑和手机。它还支持多人同时在线编辑文档&am…