luogu P2742 【模板】二维凸包
嘟嘟嘟
没错,我开始学凸包了。
其实挺简单的。
前置技能:
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 【模板】二维凸包的更多相关文章
- Luogu P2742 模板-二维凸包
Luogu P2742 模板-二维凸包 之前写的实在是太蠢了.于是重新写了一个. 用 \(Graham\) 算法求凸包. 注意两个向量 \(a\times b>0\) 的意义是 \(b\) 在 ...
- luogu P2742 【模板】二维凸包 / [USACO5.1]圈奶牛Fencing the Cows
题解: 二维凸包裸题 按照x坐标为第一关键字,y坐标为第二关键字排序 然后相邻判断叉积用单调队列搞过去 正反都做一次就好了 代码: #include <bits/stdc++.h> usi ...
- 【洛谷 P2742】【模板】二维凸包
题目链接 二维凸包板子..有时间会补总结的. #include <cstdio> #include <cmath> #include <algorithm> usi ...
- 【计算几何】二维凸包——Graham's Scan法
凸包 点集Q的凸包(convex hull)是指一个最小凸多边形,满足Q中的点或者在多边形边上或者在其内.右图中由红色线段表示的多边形就是点集Q={p0,p1,...p12}的凸包. 一组平面上的点, ...
- UVA 10652 Board Wrapping(二维凸包)
传送门 刘汝佳<算法竞赛入门经典>P272例题6包装木板 题意:有n块矩形木板,你的任务是用一个面积尽量小的凸多边形把它们抱起来,并计算出木板占整个包装面积的百分比. 输入:t组数据,每组 ...
- 计算几何 二维凸包问题 Andrew算法
凸包:把给定点包围在内部的.面积最小的凸多边形. Andrew算法是Graham算法的变种,速度更快稳定性也更好. 首先把全部点排序.依照第一keywordx第二keywordy从小到大排序,删除反复 ...
- 使用Graham扫描法求二维凸包的一个程序
#include <iostream> #include <cstring> #include <cstdlib> #include <cmath> # ...
- poj 2079 Triangle (二维凸包旋转卡壳)
Triangle Time Limit: 3000MS Memory Limit: 30000KB 64bit IO Format: %I64d & %I64u Submit Stat ...
- poj 2187 Beauty Contest(二维凸包旋转卡壳)
D - Beauty Contest Time Limit:3000MS Memory Limit:65536KB 64bit IO Format:%I64d & %I64u ...
随机推荐
- SQL 之获取DATE类的年月日语句
Select CONVERT(varchar(100), GETDATE(), 0): 05 16 2006 10:57AM Select CONVERT(varchar(100), GETDATE( ...
- 15、IO (转换流、缓冲流)
转换流概述 * A: 转换流概述 * a: 转换流概述 * OutputStreamWriter 是字符流通向字节流的桥梁:可使用指定的字符编码表,将要写入流中的字符编码成字节 * 将字符串按照指定的 ...
- Linux 中文乱码
开发中不免会接触到linux,Linux系统中文语言乱码也是我们常碰到的一个问题之一. 在网上查找了不少资料,参考了https://www.linuxidc.com/Linux/2017-07/145 ...
- MySQL常见常用的SQL优化
应尽量避免在where中使用!=或<>操作符.否则会进行全表查询 对于查询,避免全盘扫描,考虑在where或order by涉及到的列上建立索引 避免在where中进行null值判断,否则 ...
- DLL文件
Dll文件的全称是Dynamic Link Library,中文意思为动态链接库,DLL文件是不可执行文件,其是一个包含由多个程序同时使用的代码和数据的库,动态链接提供了一种方法,使进程可以调用不属于 ...
- Windows命令行方式执行OracleSQL脚本
调用格式 sqlplus user/pwd@orcl @F:\DB_BAKFile\createpro.sql>>F:\DB_BAKFile\log\createpro.log SQL脚本 ...
- OpenStack IceHouse 部署 - 1 - 架构说明
参考架构 Architecture from OpenStack Install Guide Reference Architecture Network Isolation 在本次部署中,我们采用了 ...
- sql:MySql create FUNCTION,VIEW,PROCEDURE
use geovindu; #函数 DELIMITER $$ drop function if exists f_GetDepartmentName $$ CREATE function f_GetD ...
- python_Django 实现登入功能form表单的参数接收处理
1.创建Django工程. 参考https://www.cnblogs.com/CK85/p/10159159.html中步骤. 2.在urls.py文件中添加url分发路径 "" ...
- HTML 的特殊字符转换转义符,的两种方法。
HTML 的特殊字符转换转义符,的两种方法. 方法一: function htmlEscape(str) { return String(str) .replace(/&/g, '&' ...