[SHOI2012]信用卡凸包(计算几何)
/*
考验观察法??
可以发现最终答案等于所有作为圆心的点求出凸包的周长加上一个圆的周长
向量旋转
(x1, y1) 相较于 (x2, y2) 旋转角c 答案是
(dtx * cosc - dty * sinc + x2, dty * cosc + dtx * sinc + y2)
貌似我的凸包也不鲁棒耶
*/
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<iostream>
#include<cmath>
#include<queue>
#define ll long long
#define M 400010
#define mmp make_pair
using namespace std;
int read()
{
int nm = 0, f = 1;
char c = getchar();
for(; !isdigit(c); c = getchar()) if(c == '-') f = -1;
for(; isdigit(c); c = getchar()) nm = nm * 10 + c - '0';
return nm * f;
}
const double pi = acos(-1), eps = 1e-12;
struct Vec
{
double x, y;
}pt[M];
double operator * (const Vec &a, const Vec &b)
{
return a.x * b.y - a.y * b.x;
}
Vec operator + (Vec a, Vec b)
{
return Vec{a.x + b.x, a.y + b.y};
}
Vec operator - (Vec a, Vec b)
{
return Vec{a.x - b.x, a.y - b.y};
}
double cross(Vec a, Vec b, Vec c)
{
return (a - c) * (b - c);
}
bool cmp1(Vec a, Vec b)
{
return a.y == b.y ? a.x < b.x : a.y < b.y;
}
bool cmp(Vec a, Vec b)
{
return cross(a, b, pt[1]) >= 0;
}
Vec rotate(Vec a, double p)
{
return Vec{a.x * cos(p) - a.y * sin(p), a.y * cos(p) + a.x * sin(p)};
}
double diss(Vec a, Vec b)
{
double x = a.x - b.x, y = a.y - b.y;
return sqrt(x * x + y * y);
}
double r, h, l, a, b, ap;
int n, tp, sta[M];
int main()
{
n = read();
scanf("%lf%lf%lf", &l, &h, &r);
h = (h - r * 2) / 2.0, l = (l - r * 2) / 2.0;
for(int i = 1; i <= n; i++)
{
scanf("%lf%lf%lf", &a, &b, &ap);
pt[i] = rotate(Vec{h, l}, ap) + Vec{a, b};
pt[i + 2 * n] = rotate(Vec{-h, l}, ap) + Vec{a, b};
pt[i + 3 * n] = rotate(Vec{h, -l}, ap) + Vec{a, b};
pt[i + n] = rotate(Vec{-h, -l}, ap) + Vec{a, b};
}
n *= 4;
sort(pt + 1, pt + n + 1, cmp1);
sta[++tp] = 1;
sort(pt + 2, pt + n + 1, cmp);
sta[++tp] = 2;
sta[++tp] = 3;
for(int i = 4; i <= n; i++)
{
while(tp > 1 && cross(pt[i], pt[sta[tp]], pt[sta[tp - 1]]) >= 0) tp--;
sta[++tp] = i;
}
double ans = 0;
for(int i = 1; i < tp; i++) ans += diss(pt[sta[i]], pt[sta[i + 1]]);
ans += diss(pt[sta[1]], pt[sta[tp]]);
ans += r * pi * 2;
printf("%.2lf\n", ans);
return 0;
}
[SHOI2012]信用卡凸包(计算几何)的更多相关文章
- 【BZOJ2829】[SHOI2012]信用卡凸包(凸包)
[BZOJ2829][SHOI2012]信用卡凸包(凸包) 题面 BZOJ 洛谷 题解 既然圆角的半径都是一样的,而凸包的内角和恰好为\(360°\),所以只需要把圆角的圆心弄下来跑一个凸包,再额外加 ...
- [SHOI2012]信用卡凸包(凸包+直觉)
这个题还是比较有趣. 小心发现,大胆猜想,不用证明! 我们发现所谓的信用卡凸包上弧的长度总和就是圆的周长! 然后再加上每个长宽都减去圆的直径之后的长方形的凸包周长即可! #include<ios ...
- luogu P3829 [SHOI2012]信用卡凸包 凸包 点的旋转
LINK:信用卡凸包 当 R==0的时候显然是一个点的旋转 之后再求凸包即可. 这里先说点如何旋转 如果是根据原点旋转的话 经过一个繁杂的推导可以得到一个矩阵. [cosw,-sinw] [sinw, ...
- P3829 [SHOI2012]信用卡凸包
思路 注意到结果就是每个信用卡边上的四个圆心的凸包周长+一个圆的周长 然后就好做了 注意平行时把距离小的排在前面,栈中至少要有1个元素(top>1),凸包中如果存在叉积为0的点也要pop,否则可 ...
- [洛谷P3829][SHOI2012]信用卡凸包
题目大意:有$n$张一模一样的信用卡,每个角进行了圆滑处理,问这些卡组成的“凸包”的周长 题解:发现是圆滑处理的圆心围成的凸包加上一个圆周即可 卡点:输入长宽弄反,然后以为是卡精 C++ Code: ...
- Luogu-3829 [SHOI2012]信用卡凸包
这道题的转化很巧妙,可以把信用卡四个角的圆心看做平面上的点来做凸包,\(ans\)就是凸包周长加上一个圆的周长 // luogu-judger-enable-o2 #include<cmath& ...
- BZOJ 2829 信用卡凸包 ——计算几何
凸包裸题 #include <map> #include <cmath> #include <queue> #include <cstdio> #inc ...
- 【BZOJ 2829】 2829: 信用卡凸包 (凸包)
2829: 信用卡凸包 Description Input Output Sample Input 2 6.0 2.0 0.0 0.0 0.0 0.0 2.0 -2.0 1.5707963268 Sa ...
- bzoj 2829 信用卡凸包(凸包)
2829: 信用卡凸包 Time Limit: 10 Sec Memory Limit: 128 MB Submit: 1342 Solved: 577 [Submit][Status][Disc ...
随机推荐
- ORACLE 12C 之集群日志位置变化
如果你还是使用 oracle 11g RAC 的目录结构方式寻找集群的日志,你会发现目录中所有的日志都是空的.actdb21:/oracle/app/12.2.0/grid/log/actdb21(+ ...
- 3、eclipse集成svn
手动安装 1.从官网下载适合自己开发工具的site-.zip文件,网址是:subclipse.tigris.org,2.从中解压出features与plugins文件夹,复制到eclipse的目录my ...
- 亿级用户百TB级数据的AIOps 技术实践之路
关于面临的挑战 "因为专业性强,我认为反而让交互方式变简单了,打个点餐的比方,软件1.0阶段是,我要吃鱼香肉丝,我要吃辣的或是素一点的,根据清晰的接口上菜.而软件2.0阶段就是,我今天想吃开 ...
- java中三种for循环之间的对比
普通for循环语法: for (int i = 0; i < integers.length; i++) { System.out.println(intergers[i]); } foreac ...
- Feign 使用入门
Feign 的目的是简化 Web Service 客户端的开发,在使用 Feign 时,使用注解来修饰接口,被注解修饰的接口具有访问 Web Service 的能力,包括 Feign 自带的注解,也支 ...
- Python实战(6)单线程和多线程导入mysql数据对比测试
单线程脚本 导入文件的行数 # wc -l /data/logs/testlog/20120219/testlog1/* 1510503 total # -*- coding: utf-8 -*- # ...
- ubuntu-docker入门到放弃(六)数据管理
在docker的数据管理中,有两个概念: 1.数据卷 数据卷是一个可供容器使用的特殊目录,它绕过文件系统,可以提供很多有用的特性: 1.1 数据卷可以在容器之间共享和重用 1.2 对数据卷的修改会立刻 ...
- Windows Azure Virtual Network (12) 虚拟网络之间点对点连接VNet Peering
<Windows Azure Platform 系列文章目录> 在有些时候,我们需要通过VNet Peering,把两个虚拟网络通过内网互通互联.比如: 1.在订阅A里的Virtual N ...
- 【Git】Git使用--常用命令
查看所有分支 git branch -a 查看本地分支 git branch 切换分支 git checkout test demo git checkout release_1.3.1 (切换到re ...
- uoj #14.【UER #1】DZY Loves Graph
http://uoj.ac/problem/14 由于加入的边权递增,可以直接运行kruskal并支持撤销,但这样如果反复批量删边和撤销,时间复杂度会退化,因此需要对删边操作加上延时处理,只有在删边后 ...