原文链接:http://www.cnblogs.com/devymex/archive/2010/08/09/1795392.html

C++/STL实现:

#include <algorithm>
#include <iostream>
#include <vector>
#include <math.h>
using namespace std;
//二维点(或向量)结构体定义
#ifndef _WINDEF_
struct POINT
{
int x;
int y;
};
#endif
typedef vector<POINT> PTARRAY;
//判断两个点(或向量)是否相等
bool operator==(const POINT &pt1, const POINT &pt2)
{
return (pt1.x == pt2.x && pt1.y == pt2.y);
}
// 比较向量中哪个与x轴向量(1, 0)的夹角更大
bool CompareVector(const POINT &pt1, const POINT &pt2)
{
//求向量的模
float m1 = sqrt((float)(pt1.x * pt1.x + pt1.y * pt1.y));
float m2 = sqrt((float)(pt2.x * pt2.x + pt2.y * pt2.y));
//两个向量分别与(1, 0)求内积
float v1 = pt1.x / m1, v2 = pt2.x / m2;
//如果向量夹角相等,则返回离基点较近的一个,保证有序
return (v1 > v2 || v1 == v2 && m1 < m2);
}
//计算凸包
void CalcConvexHull(PTARRAY &vecSrc)
{
//点集中至少应有3个点,才能构成多边形
if (vecSrc.size() < )
{
return;
}
//查找基点
POINT ptBase = vecSrc.front(); //将第1个点预设为最小点
for (PTARRAY::iterator i = vecSrc.begin() + ; i != vecSrc.end(); ++i)
{
//如果当前点的y值小于最小点,或y值相等,x值较小
if (i->y < ptBase.y || (i->y == ptBase.y && i->x > ptBase.x))
{
//将当前点作为最小点
ptBase = *i;
}
}
//计算出各点与基点构成的向量
for (PTARRAY::iterator i = vecSrc.begin(); i != vecSrc.end();)
{
//排除与基点相同的点,避免后面的排序计算中出现除0错误
if (*i == ptBase)
{
i = vecSrc.erase(i);
}
else
{
//方向由基点到目标点
i->x -= ptBase.x, i->y -= ptBase.y;
++i;
}
}
//按各向量与横坐标之间的夹角排序
sort(vecSrc.begin(), vecSrc.end(), &CompareVector);
//删除相同的向量
vecSrc.erase(unique(vecSrc.begin(), vecSrc.end()), vecSrc.end());
//计算得到首尾依次相联的向量
for (PTARRAY::reverse_iterator ri = vecSrc.rbegin();
ri != vecSrc.rend() - ; ++ri)
{
PTARRAY::reverse_iterator riNext = ri + ;
//向量三角形计算公式
ri->x -= riNext->x, ri->y -= riNext->y;
}
//依次删除不在凸包上的向量
for (PTARRAY::iterator i = vecSrc.begin() + ; i != vecSrc.end(); ++i)
{
//回溯删除旋转方向相反的向量,使用外积判断旋转方向
for (PTARRAY::iterator iLast = i - ; iLast != vecSrc.begin();)
{
int v1 = i->x * iLast->y, v2 = i->y * iLast->x;
//如果叉积小于0,则无没有逆向旋转
//如果叉积等于0,还需判断方向是否相逆
if (v1 < v2 || (v1 == v2 && i->x * iLast->x > &&
i->y * iLast->y > ))
{
break;
}
//删除前一个向量后,需更新当前向量,与前面的向量首尾相连
//向量三角形计算公式
i->x += iLast->x, i->y += iLast->y;
iLast = (i = vecSrc.erase(iLast)) - ;
}
}
//将所有首尾相连的向量依次累加,换算成坐标
vecSrc.front().x += ptBase.x, vecSrc.front().y += ptBase.y;
for (PTARRAY::iterator i = vecSrc.begin() + ; i != vecSrc.end(); ++i)
{
i->x += (i - )->x, i->y += (i - )->y;
}
//添加基点,全部的凸包计算完成
vecSrc.push_back(ptBase);
} int main(void)
{
int nPtCnt = ; //生成的随机点数
PTARRAY vecSrc, vecCH;
for (int i = ; i < nPtCnt; ++i)
{
POINT ptIn = { rand() % , rand() % };
vecSrc.push_back(ptIn);
cout << ptIn.x << ", " << ptIn.y << endl;
}
CalcConvexHull(vecSrc);
cout << "\nConvex Hull:\n";
for (PTARRAY::iterator i = vecSrc.begin(); i != vecSrc.end(); ++i)
{
cout << i->x << ", " << i->y << endl;
}
return ;
}

Graham's Scan算法的更多相关文章

  1. 计算几何---凸包问题(Graham/Andrew Scan )

    概念 凸包(Convex Hull)是一个计算几何(图形学)中的概念.用不严谨的话来讲,给定二维平面上的点集,凸包就是将最外层的点连接起来构成的凸多边型,它能包含点集中所有点的.严谨的定义和相关概念参 ...

  2. 凸包Graham Scan算法实现

    凸包算法实现点集合中搜索凸包顶点的功能,可以处理共线情况,可以输出共线点也可以不输出而只输出凸包顶点.经典的Graham Scan算法,点排序使用极角排序方式,并对共线情况做特殊处理.一般算法是将共线 ...

  3. 【计算几何】二维凸包——Graham's Scan法

    凸包 点集Q的凸包(convex hull)是指一个最小凸多边形,满足Q中的点或者在多边形边上或者在其内.右图中由红色线段表示的多边形就是点集Q={p0,p1,...p12}的凸包. 一组平面上的点, ...

  4. 磁盘寻道时间算法之----------------SCAN算法和最短寻道时间优先调度算法

    若干个等待访问磁盘者依次要访问的柱面编号为:80,40,74,14,60,31,61,假设每移动一个柱面需要4毫秒时间,移动到当前位于35号柱面,且当前的移动方向向柱面号增加的方向.请计算: (1)若 ...

  5. Beauty Contest(graham求凸包算法)

    Time Limit: 3000MS   Memory Limit: 65536K Total Submissions: 25256   Accepted: 7756 Description Bess ...

  6. Graham's Scan法求解凸包问题

    概念 凸包(Convex Hull)是一个计算几何(图形学)中的概念.用不严谨的话来讲,给定二维平面上的点集,凸包就是将最外层的点连接起来构成的凸多边型,它能包含点集中所有点的.严谨的定义和相关概念参 ...

  7. DB Scan算法的分析与实现

    摘自:http://www.cnblogs.com/weixliu/archive/2012/12/08/2808815.html 根据上面第二个数据集的簇的形状比较怪异,分簇结果应该是连起来的属于一 ...

  8. 电梯调度算法---SCAN算法

    请珍惜小编劳动成果,该文章为小编原创,转载请注明出处. 扫描(SCAN)调度算法:总是从磁臂当前位置开始,沿磁臂的移动方向去选择离当前磁臂最近的那个柱面的访问者.如果沿磁臂的方向无请求访问时,就改变磁 ...

  9. Hello World 之 CGAL

    CGAL有神秘的面纱,让我不断想看清其真面目.开始吧! 1 Three Points and One Segment 第一个例子是创建3个点和一条线段,并且在其上进行一些操作. 所有的CGAL头文件都 ...

随机推荐

  1. ThinkPHP框架概述

    框架概述 1.什么是框架 框架,即framework.其实就是某种应用的半成品,就是一组组件,供你选用完成你自己的系统. 2.框架的应用场景 ① 代码重用 ② 合理分工 ③ 解决团队协作开发问题 3. ...

  2. Standford CoreNLP

    Stanford CoreNLP Stanford CoreNLP提供一组自然语言处理的工具.这些工具可以把原始英语文本作为输入,输出词的基本形式,词的词性标记,判断词是否是公司名.人名等,规格化日期 ...

  3. PHP 正则表达式替换一部分内容

    preg_replace('/&topic=(.*?)&type=/',"&topic={$data['topic']}&type=",$postF ...

  4. 清空系统日志shell scripts——自学笔记

    这是一个清空系统日志的脚本: vim logmess_clean.sh #bin/bash        //该脚本所使用的shell解释器 cd /var/log/  //切换到存放日志目录 ech ...

  5. Ping N个IP测试网络连通性

    #-----------------------Smokeping移动节点-------------------##! /bin/bashecho "------------- Statin ...

  6. winform Config文件操作

    using System;using System.Collections.Generic;using System.Text;using System.Xml;using System.Config ...

  7. Spring-Mybatis 异常记录(1)

    Spring  applicationconfig.xml如下 <?xml version="1.0" encoding="UTF-8"?> < ...

  8. Android砖机救活(索爱MT15i)

    前言 接触Android时间长了就想编译一套属于自己的系统,摘取不必要的那些组件,然后刷到手机上,俗话说的好,“常在河 边走,哪有不湿鞋”.果不其然,刷完自己编译的系统手机变砖了,具体情况为 开不开机 ...

  9. 【PHP】phpcms html去除空白

    // 文件路径:/phpcms/libs/classes/template_cache.class.php 42行 // 第四第五行是新增的 $content = $this->template ...

  10. Sending Email from mailx Command in Linux Using Gmail’s SMTP

    The mailx or mail command in Linux is still providing service for guys like me, especially when we n ...