#include"stdafx.h"
#include<iostream>
#include<cmath>
#define TRUE 1
#define FALSE 0
using namespace std;
typedef struct Node//坐标点
{
 double x;
 double y;
}Node;  
typedef struct List
{
 Node* data;      //点
 int count;      //点的个数
}List;
typedef struct CloseNode
{
 Node a;
 Node b;     //计算距离的两个点
 double space;     //距离平方
}CloseNode;
int n;     //点的数目
//输入各点到List中
void create(List &L)
{
 cout << "请输入平面上点的数目:\n";
 cin >> n;
 L.count = n;
 L.data = new Node[L.count];      //动态空间分配
 cout << "输入各点坐标 :x_y):" << endl;
 for (int i = 0; i<L.count; ++i)
  cin >> L.data[i].x >> L.data[i].y;
}
//求距离的平方
double square(Node a, Node b)
{
 return ((a.x - b.x)*(a.x - b.x)) + ((a.y - b.y)*(a.y - b.y));
}
//冒泡排序
void BubbleSort(Node r[], int length)
{
 int change, n;
 n = length; change = TRUE;
 double b, c;
 for (int i = 0; i<n - 1 && change; ++i)
 {
  change = FALSE;
  for (int j = 0; j<n - i - 1; ++j)
  {
   if (r[j].x>r[j + 1].x)
   {
    b = r[j].x; c = r[j].y;
    r[j].x = r[j + 1].x; r[j].y = r[j + 1].y;
    r[j + 1].x = b; r[j + 1].y = c;
    change = TRUE;
   }
  }
 }
}
//分治法中先将坐标按X轴从小到大的顺序排列
void paixu(List L)
{
 BubbleSort(L.data, L.count);   //调用冒泡排序
}
//左右各距中线d的区域的最近对算法
void middle(const List & L, CloseNode &cnode, int mid, double midX)
{
 int i, j;    //分别表示中线左边,右边的点
 double d = sqrt(cnode.space);
 i = mid;
 while (i >= 0 && L.data[i].x >= (midX - d))    //在左边的d区域内
 {
  j = mid;
  while (L.data[++j].x <= (midX + d) && j <= L.count)    //在右边的d区域内
  {
   if (L.data[j].y<(L.data[i].y - d) || L.data[j].y>(L.data[i].y + d))   //判断纵坐标是否在左边某固定点的2d区域内
    continue;
   double space = square(L.data[i], L.data[j]);
   if (cnode.space>space)    //在满足条件的区域内依次判断
   {
    cnode.a = L.data[i];
    cnode.b = L.data[j];
    cnode.space = space;
   }
  }
  --i;
 }
}
//分治法求最近对
void DivideConquer(const List &L, CloseNode &closenode, int begin, int end)
{
 if (begin != end)
 {
  int mid = (begin + end) / 2;     //排列后的中间的那个点
  double midX = L.data[mid].x;
  DivideConquer(L, closenode, begin, mid);      //继续在左半边用分治法求最近对
  DivideConquer(L, closenode, mid + 1, end);      //继续在右半边用分治法求最近对
  middle(L, closenode, mid, midX);               //判断左右各距中线d的区域,是否有最近对
 }
}
void main()
{
 //初始化
 List list;
 CloseNode closenode;
 closenode.space = 10000;    
 create(list);    
 cout << "各点坐标为:" << endl;
 for (int i = 0; i<list.count; ++i)
  cout << "X=" << list.data[i].x << "   Y=" << list.data[i].y << "\n";
 cout << "用分治法求最近对:" << endl;
 paixu(list);
 cout << "经过排序后的各点:" << endl;
 for (int j = 0; j<list.count; ++j)
  cout << "X=" << list.data[j].x << "   Y=" << list.data[j].y << "\n";
 DivideConquer(list, closenode, 0, list.count - 1);
 cout << "最近对为点 (" << closenode.a.x << "," << closenode.a.y << ")和点(" << closenode.b.x << "," << closenode.b.y << ")\n" << "最近距离为: " << sqrt(closenode.space) << endl;
}
 

分治法求解最近对问题(c++)的更多相关文章

  1. [C++] 分治法之棋盘覆盖、循环赛日程表

    一.分治的基本思想 将一个难以直接解决的大问题,分割成一些规模较小的相同问题,以便各个击破,分而治之. 对于一个规模为 n 的问题,若问题可以容易地解决,则直接解决,否则将其分解为 k 个规模较小的子 ...

  2. Java算法——分治法

         一.基本概念 在计算机科学中,分治法是一种很重要的算法.字面上的解释是“分而治之”,就是把一个复杂的问题分成两个或更多的相同或相似的子问题,再把子问题分成更小的子问题……直到最后子问题可以简 ...

  3. 分治法避免定义多个递归函数,应该使用ResultType

    总结:对二叉树应用分治法时,应避免定义多个递归函数,当出现需要递归求解多种的结果时,尽量使用ResultType来让一次递归返回多种结果. 题目:Binary Tree Maximum Path Su ...

  4. 分治法(一)(zt)

    这篇文章将讨论: 1) 分治策略的思想和理论 2) 几个分治策略的例子:合并排序,快速排序,折半查找,二叉遍历树及其相关特性. 说明:这几个例子在前面都写过了,这里又拿出来,从算法设计的策略的角度把它 ...

  5. p1257 平面上最接近点对---(分治法)

    首先就是一维最接近点的情况... #include<iostream> #include<cstdio> #include<cstring> #include< ...

  6. 分治法及其python实现例子

    在前面的排序算法学习中,归并排序和快速排序就是用的分治法,分治法作为三大算法之一的,有非常多的应用例子. 分治法概念 将一个复杂的问题分成两个或更多的相同或相似的子问题,再把子问题分成更小的子问题-- ...

  7. 分治法 - Divide and Conquer

    在计算机科学中,分治法是一种很重要的算法.分治法即『分而治之』,把一个复杂的问题分成两个或更多的相同或相似的子问题,再把子问题分成更小的子问题……直到最后子问题可以简单的直接求解,原问题的解即子问题的 ...

  8. poj 3714 Raid【(暴力+剪枝) || (分治法+剪枝)】

    题目:  http://poj.org/problem?id=3714 http://acm.hust.edu.cn/vjudge/contest/view.action?cid=27048#prob ...

  9. python编写PAT 1007 Maximum Subsequence Sum(暴力 分治法 动态规划)

    python编写PAT甲级 1007 Maximum Subsequence Sum wenzongxiao1996 2019.4.3 题目 Given a sequence of K integer ...

随机推荐

  1. EntityFramework Core Raw SQL

    前言 本节我们来讲讲EF Core中的原始查询,目前在项目中对于简单的查询直接通过EF就可以解决,但是涉及到多表查询时为了一步到位就采用了原始查询的方式进行.下面我们一起来看看. EntityFram ...

  2. 前端CSS预处理器Sass

    前面的话   "CSS预处理器"(css preprocessor)的基本思想是,用一种专门的编程语言,进行网页样式设计,然后再编译成正常的CSS文件.SASS是一种CSS的开发工 ...

  3. 【java】Naming.bind和Registry.bind区别

    Naming类和Registry类均在java.rmi包 Naming类通过解析URI绑定远程对象,将URI拆分成主机.端口和远程对象名称,使用的仍是Registry类. public static ...

  4. 多线程的通信和同步(Java并发编程的艺术--笔记)

    1. 线程间的通信机制 线程之间通信机制有两种: 共享内存.消息传递.   2. Java并发 Java的并发采用的是共享内存模型,Java线程之间的通信总是隐式执行,通信的过程对于程序员来说是完全透 ...

  5. PowerDesigner-VBSrcipt-自动设置主键,外键名等(SQL Server)

    在PowerDesigner中的设计SQL Server 数据表时,要求通过vbScript脚本实现下面的功能: 主键:pk_TableName 外键:fk_TableName_ForeignKeyC ...

  6. Zabbix基本配置及监控主机

    监控主机一版需要在被监控的主机上安装Zabbix Agent 监控主机 安装zabbix-agent 首先需要在被监控的主机上安装agent,可以下载预编译好的RPM进行安装,下载地址:http:// ...

  7. 从零开始编写自己的C#框架(26)——小结

    一直想写个总结,不过实在太忙了,所以一直拖啊拖啊,拖到现在,不过也好,有了这段时间的沉淀,发现自己又有了小小的进步.哈哈...... 原想框架开发的相关开发步骤.文档.代码.功能.部署等都简单的讲过了 ...

  8. 深入node之Transform

    Transform流特性 在开发中直接接触Transform流的情况不是很多,往往是使用相对成熟的模块或者封装的API来完成流的处理,最为特殊的莫过于through2模块和gulp流操作.那么,Tra ...

  9. ubuntu 下安装scrapy

    1.把Scrapy签名的GPG密钥添加到APT的钥匙环中: sudo apt-key adv --keyserver hkp://keyserver.ubuntu.com:80 --recv 6272 ...

  10. 【SAP业务模式】之ICS(七):IDOC配置

    这是ICS业务模式系列的最后一篇了,主要讲解IDOC的配置. 一.指定EDI传输的供应商逻辑地址 事务代码:WEL1 注意:上面逻辑地址是生产公司+内部客户.有以下两种情形: 1.如果内部客户都是纯数 ...