嘟嘟嘟




没错,我开始学凸包了。

其实挺简单的。




前置技能:

1.极坐标系

2.向量叉积




1.极坐标系

就是一种二维坐标系。只不过两个坐标分别表示向量和极轴的角度和自身的长度。对于不同的问题,极轴可以自己选取。

2.向量叉积

不说了




算法是\(Graham\)扫描法,下面讲一下实现步骤:

1.在所有点中找到横坐标最小的点作为极点,如果有多个,取纵坐标最小的点。

2.对于其他\(n -1\)个点进行极角排序,极角相同比较到极点距离。

排完序后的图大概是这个样子的:



其中的字母就是排完序后的序号。

3.然后维护一个栈,对于每一个点,比较他和栈顶的两个点构成的线段,如果在线段的右边,就把栈顶弹出,直到栈中只剩一个元素或该点在线段右边为止。

4.最后栈中的元素就是凸包的顶点。




再补充几点:

1.极角排序的函数怎么写。

对于极点\(A\)和要排序的两个点\(B, C\)。就是看\(C\)在\(AB\)的上方还是下方。这个用叉积判断即可。如果\(\overrightarrow{AB} \times \overrightarrow{AC} < 0\),说明\(C\)在\(AB\)下方,反之亦然。

如果叉积等于\(0\),比较到\(A\)点距离。

2.判断点\(i\)在栈顶两个元素所成直线的左右。

跟上面一样,叉积。

#include<cstdio>
#include<iostream>
#include<cmath>
#include<algorithm>
#include<cstring>
#include<cstdlib>
#include<cctype>
#include<vector>
#include<stack>
#include<queue>
using namespace std;
#define enter puts("")
#define space putchar(' ')
#define Mem(a, x) memset(a, x, sizeof(a))
#define rg register
typedef long long ll;
typedef double db;
const int INF = 0x3f3f3f3f;
const db eps = 1e-8;
const int maxn = 1e4 + 5;
inline ll read()
{
ll ans = 0;
char ch = getchar(), last = ' ';
while(!isdigit(ch)) last = ch, ch = getchar();
while(isdigit(ch)) ans = (ans << 1) + (ans << 3) + ch - '0', ch = getchar();
if(last == '-') ans = -ans;
return ans;
}
inline void write(ll x)
{
if(x < 0) x = -x, putchar('-');
if(x >= 10) write(x / 10);
putchar(x % 10 + '0');
} int n;
struct Vec
{
db x, y;
db operator * (const Vec& oth)const
{
return x * oth.y - oth.x * y;
}
friend inline db dis(const Vec& A)
{
return A.x * A.x + A.y * A.y;
}
};
struct Point
{
db x, y;
int id;
Vec operator - (const Point& oth)const
{
return (Vec){x - oth.x, y - oth.y};
}
friend void swap(Point& A, Point& B)
{
swap(A.x, B.x); swap(A.y, B.y);
}
}p[maxn], S;
inline bool cmp(Point A, Point B)
{
db s = (A - S) * (B - S);
if(fabs(s) > eps) return s > eps;
return dis(A - S) - dis(B - S) < -eps;
} db solve(Point A, Point B, Point C)
{
return (B - A) * (C - A);
} void init()
{
int id = 1;
for(int i = 2; i <= n; ++i)
if(p[i].x < p[id].x - eps || (fabs(p[i].x - p[id].x) < eps && p[i].y < p[id].y)) id = i;
if(id != 1) swap(p[1], p[id]);
S.x = p[1].x; S.y = p[1].y;
for(int i = 1; i <= n; ++i) p[i].id = i;
sort(p + 2, p + n + 1, cmp);
} int st[maxn], top = 0;
db ans = 0; int main()
{
n = read();
for(int i = 1; i <= n; ++i) scanf("%lf%lf", &p[i].x, &p[i].y);
init();
st[++top] = 1;
for(int i = 2; i <= n; ++i)
{
while(top > 1 && solve(p[st[top - 1]], p[st[top]], p[i]) < -eps) top--;
st[++top] = i;
}
st[top + 1] = st[1];
for(int i = 1; i <= top; ++i)
ans += sqrt(dis(p[st[i + 1]] - p[st[i]]));
printf("%.2lf\n", ans);
return 0;
}

然后我因为极点没赋初值\(Debug\)了不知多长时间……

luogu P2742 【模板】二维凸包的更多相关文章

  1. Luogu P2742 模板-二维凸包

    Luogu P2742 模板-二维凸包 之前写的实在是太蠢了.于是重新写了一个. 用 \(Graham\) 算法求凸包. 注意两个向量 \(a\times b>0\) 的意义是 \(b\) 在 ...

  2. luogu P2742 【模板】二维凸包 / [USACO5.1]圈奶牛Fencing the Cows

    题解: 二维凸包裸题 按照x坐标为第一关键字,y坐标为第二关键字排序 然后相邻判断叉积用单调队列搞过去 正反都做一次就好了 代码: #include <bits/stdc++.h> usi ...

  3. 【洛谷 P2742】【模板】二维凸包

    题目链接 二维凸包板子..有时间会补总结的. #include <cstdio> #include <cmath> #include <algorithm> usi ...

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

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

  5. UVA 10652 Board Wrapping(二维凸包)

    传送门 刘汝佳<算法竞赛入门经典>P272例题6包装木板 题意:有n块矩形木板,你的任务是用一个面积尽量小的凸多边形把它们抱起来,并计算出木板占整个包装面积的百分比. 输入:t组数据,每组 ...

  6. 计算几何 二维凸包问题 Andrew算法

    凸包:把给定点包围在内部的.面积最小的凸多边形. Andrew算法是Graham算法的变种,速度更快稳定性也更好. 首先把全部点排序.依照第一keywordx第二keywordy从小到大排序,删除反复 ...

  7. 使用Graham扫描法求二维凸包的一个程序

    #include <iostream> #include <cstring> #include <cstdlib> #include <cmath> # ...

  8. poj 2079 Triangle (二维凸包旋转卡壳)

    Triangle Time Limit: 3000MS   Memory Limit: 30000KB   64bit IO Format: %I64d & %I64u Submit Stat ...

  9. poj 2187 Beauty Contest(二维凸包旋转卡壳)

    D - Beauty Contest Time Limit:3000MS     Memory Limit:65536KB     64bit IO Format:%I64d & %I64u ...

随机推荐

  1. SQL 之获取DATE类的年月日语句

    Select CONVERT(varchar(100), GETDATE(), 0): 05 16 2006 10:57AM Select CONVERT(varchar(100), GETDATE( ...

  2. 15、IO (转换流、缓冲流)

    转换流概述 * A: 转换流概述 * a: 转换流概述 * OutputStreamWriter 是字符流通向字节流的桥梁:可使用指定的字符编码表,将要写入流中的字符编码成字节 * 将字符串按照指定的 ...

  3. Linux 中文乱码

    开发中不免会接触到linux,Linux系统中文语言乱码也是我们常碰到的一个问题之一. 在网上查找了不少资料,参考了https://www.linuxidc.com/Linux/2017-07/145 ...

  4. MySQL常见常用的SQL优化

    应尽量避免在where中使用!=或<>操作符.否则会进行全表查询 对于查询,避免全盘扫描,考虑在where或order by涉及到的列上建立索引 避免在where中进行null值判断,否则 ...

  5. DLL文件

    Dll文件的全称是Dynamic Link Library,中文意思为动态链接库,DLL文件是不可执行文件,其是一个包含由多个程序同时使用的代码和数据的库,动态链接提供了一种方法,使进程可以调用不属于 ...

  6. Windows命令行方式执行OracleSQL脚本

    调用格式 sqlplus user/pwd@orcl @F:\DB_BAKFile\createpro.sql>>F:\DB_BAKFile\log\createpro.log SQL脚本 ...

  7. OpenStack IceHouse 部署 - 1 - 架构说明

    参考架构 Architecture from OpenStack Install Guide Reference Architecture Network Isolation 在本次部署中,我们采用了 ...

  8. sql:MySql create FUNCTION,VIEW,PROCEDURE

    use geovindu; #函数 DELIMITER $$ drop function if exists f_GetDepartmentName $$ CREATE function f_GetD ...

  9. python_Django 实现登入功能form表单的参数接收处理

    1.创建Django工程. 参考https://www.cnblogs.com/CK85/p/10159159.html中步骤. 2.在urls.py文件中添加url分发路径 "" ...

  10. HTML 的特殊字符转换转义符,的两种方法。

    HTML 的特殊字符转换转义符,的两种方法. 方法一: function htmlEscape(str) { return String(str) .replace(/&/g, '&' ...