[CQOI2006]凸多边形(半平面交)
很明显是一道半平面交的题。
先说一下半平面交的步骤:
1.用点向法(点+向量)表示直线
2.极角排序,若极角相同,按相对位置排序。
3.去重,极角相同的保留更优的
4.枚举边维护双端队列
5.求答案
1就不说了,2中的极角可以用atan2(y,x)来求,因为atan2精度要高

双端队列的原因是新加的一条边对头和尾都有影响,如图:

如何去判断:只要判断线head和线head+1,的交点p与新的一条线的位置关系就可以
至于交点的求法:先见图:

求\(p_1v_1,p_2v_2\)的交点\(p_0\)
设\(p_0=p_2+kv_2\ \ u=p_2-p_1\)
\(S_1=u\times v_1,S_2=v_1\times v_2,k=S_1/S2\)
所以\(p_0=p_2+kv_2\)
\(S_1\)为\(u\)与\(v_1\)的面积,\(S_2\)为\(v_1\)与\(v_2\)的面积,按比例求得\(k\)再乘一下就求出\(p_0\)
最后统计答案
细节见代码:
#include<iostream>
#include<cstdio>
#include<string>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<queue>
#include<stack>
#include<set>
#include<bitset>
#include<vector>
#include<cstdlib>
#define QAQ int
#define TAT long long
#define OwO bool
#define ORZ double
#define F(i,j,n) for(QAQ i=j;i<=n;++i)
#define E(i,j,n) for(QAQ i=j;i>=n;--i)
#define MES(i,j) memset(i,j,sizeof(i))
#define MEC(i,j) memcpy(i,j,sizeof(j))
using namespace std;
const int N=505;
const double eps=1e-8;
int n;
struct Point{
double x,y;
friend Point operator - (Point a,Point b){
Point t;
t.x=a.x-b.x;t.y=a.y-b.y;
return t;
}
friend Point operator + (Point a,Point b){
Point t;
t.x=a.x+b.x;t.y=a.y+b.y;
return t;
}
friend double operator * (Point a,Point b){
return a.x*b.x+a.y*b.y;
}
friend double operator ^ (Point a,Point b){
return a.x*b.y-b.x*a.y;
}
friend Point operator * (double k,Point b){
Point t;
t.x=k*b.x;t.y=k*b.y;
return t;
}
}b[N];
int sign(double x){
return fabs(x)<=eps ? 0 : (x>0 ? 1 : -1);
}
struct Line{
Point p,v;
double poa;
friend OwO operator < (Line x,Line y){
return sign(x.poa-y.poa)==0 ? sign((x.v-x.p) ^ (y.v-x.p)) >0 : sign(x.poa-y.poa)<0;
//因为我是向量左侧求交,所以极角相同时靠左的更优,把优的放在后面,方便之后的操作 ,可以画图体会一下
}
}a[N],q[N];
int js,cnt,head,tail;
double ans;
Point inter(Line a,Line b){//求交点
Point p1=a.p,p2=b.p,v1=a.v,v2=b.v;
v1=v1-p1;v2=v2-p2;
Point u=p2-p1;
Point p=p2+((u^v1)/(v1^v2))*v2;
return p;
}
OwO pd(Line i,Line j,Line k){
Point p=inter(i,j);
return sign((k.v-k.p) ^ (p-k.p))<0;
}
void Half_Plane(){
sort(a+1,a+js+1);//排序
F(i,1,js) {
if(sign(a[i].poa-a[i-1].poa)!=0) cnt++;
a[cnt]=a[i];//因为排过序,即使极角相同,后面的也比前面的优
}
head=1;tail=0;
q[++tail]=a[1];q[++tail]=a[2];
F(i,3,cnt){
while(head<tail&&pd(q[tail-1],q[tail],a[i])) tail--;//维护双端队列
while(head<tail&&pd(q[head+1],q[head],a[i])) head++;
q[++tail]=a[i];
}
while(head<tail&&pd(q[tail-1],q[tail],q[head])) tail--;
while(head<tail&&pd(q[head+1],q[head],q[tail])) head++;
q[tail+1]=q[head];
js=0;
F(i,head,tail) b[++js]=inter(q[i],q[i+1]);
}
int main(){
scanf("%d",&n);
F(i,1,n){
int k;
scanf("%d",&k);
F(j,1,k) scanf("%lf%lf",&b[j].x,&b[j].y);
b[k+1]=b[1];
F(j,1,k) a[++js].p=b[j],a[js].v=b[j+1];
}
F(i,1,js) a[i].poa=atan2(a[i].v.y-a[i].p.y,a[i].v.x-a[i].p.x);
Half_Plane();
b[js+1]=b[1];
if(js>2) F(i,1,js) ans+=(b[i]^b[i+1]);
ans=fabs(ans)/2.0;
printf("%.3lf\n",ans);
return 0;
}
[CQOI2006]凸多边形(半平面交)的更多相关文章
- bzoj 2618: [Cqoi2006]凸多边形 [半平面交]
2618: [Cqoi2006]凸多边形 半平面交 注意一开始多边形边界不要太大... #include <iostream> #include <cstdio> #inclu ...
- 洛谷 P4196 [CQOI2006]凸多边形 (半平面交)
题目链接:P4196 [CQOI2006]凸多边形 题意 给定 \(n\) 个凸多边形,求它们相交的面积. 思路 半平面交 半平面交的模板题. 代码 #include <bits/stdc++. ...
- BZOJ - 2618 凸多边形 (半平面交)
题意:求n个凸多边形的交面积. 半平面交模板题. #include<bits/stdc++.h> using namespace std; typedef long long ll; ty ...
- 【BZOJ 2618】 2618: [Cqoi2006]凸多边形 (半平面交)
2618: [Cqoi2006]凸多边形 Description 逆时针给出n个凸多边形的顶点坐标,求它们交的面积.例如n=2时,两个凸多边形如下图: 则相交部分的面积为5.233. Input 第一 ...
- bzoj 2618 2618: [Cqoi2006]凸多边形(半平面交)
2618: [Cqoi2006]凸多边形 Time Limit: 5 Sec Memory Limit: 128 MBSubmit: 656 Solved: 340[Submit][Status] ...
- 【BZOJ2618】[CQOI2006]凸多边形(半平面交)
[BZOJ2618][CQOI2006]凸多边形(半平面交) 题面 BZOJ 洛谷 题解 这个东西就是要求凸多边形的边所形成的半平面交. 那么就是一个半平面交模板题了. 这里写的是平方的做法. #in ...
- 2018.07.04 BZOJ 2618 Cqoi2006凸多边形(半平面交)
2618: [Cqoi2006]凸多边形 Time Limit: 5 Sec Memory Limit: 128 MB Description 逆时针给出n个凸多边形的顶点坐标,求它们交的面积.例如n ...
- 【BZOJ-2618】凸多边形 计算几何 + 半平面交 + 增量法 + 三角剖分
2618: [Cqoi2006]凸多边形 Time Limit: 5 Sec Memory Limit: 128 MBSubmit: 959 Solved: 489[Submit][Status] ...
- 算法复习——半平面交(bzoj2618凸多边形)
讲解: 这里套用wuvin神犇的ppt,附上友情链接:http://blog.leanote.com/wuvin 半平面交: 算法流程: 注意事项: 例题: Description 逆时针给出n个凸多 ...
随机推荐
- Element-UI使用基本介绍
Element-Ul是饿了么前端团队推出的一款基于Vue.js 2.0 的桌面端UI框架,手机端有对应框架是Mint UI . 开发环境 1.安装webpack npm install -g webp ...
- 斯坦福CS229机器学习课程笔记 part2:分类和逻辑回归 Classificatiion and logistic regression
Logistic Regression 逻辑回归 1.模型 逻辑回归解决的是分类问题,并且是二元分类问题(binary classification),y只有0,1两个取值.对于分类问题使用线性回归不 ...
- 缓存数据库redis
什么是Redis? Redis是一个TCP服务器,支持请求/响应协议. 在Redis中,请求通过以下步骤完成: 客户端向服务器发送查询,并从套接字读取,通常以阻塞的方式,用于服务器响应. 服务器处理命 ...
- 查询sqlserver数据库视图、存储过程等包含特定的字符串
SELECT A.name , B.definition FROM SYS.objects A INNER JOIN sys.sql_modules B ON A.object_id = B.obje ...
- Android访问中央气象台的天气预报API得到天气数据
最新说明:该接口已失效! 2014-03-04 可申请它公布的API,需申请:http://smart.weather.com.cn/wzfw/smart/weatherapi.shtml 在用A ...
- 虚拟机安装CentOS以及SecureCRT设置【完美无错版】
一.CentOS简介 CentOS是Linux的发行版之一,它安全.稳定.高效,是我最喜欢的Linux发行版之一.CentOS根据Red Hat Enterprise Linux开放源代码编译而成,与 ...
- YII2 模型关联之 一对多
需求,一个用户有多篇文章全部查询出来 文章表 用户表 //首先查找出一个用户出来 $user=Users::find()->'])->one(); //第一个参数还是关联的模型,第二个依旧 ...
- php const static define 基本用法和区别
const 定义一些在运行时不能被改变的数值.一旦值被改变,则会发生错误. 特性 只能用类名访问.不需要用 $ 符号 <?php class test{ const pi=123.12321 ...
- 【Docker官方文档】理解Docker
本文来自Docker的官方文档,详细介绍了Docker的体系结构.重要概念.内部工作机理等内容,推荐不了解Docker内部原理的同学阅读. 什么是Docker? Docker是一个用于开发.交付和运行 ...
- Linux extundelete命令
一.简介 extundelete的文件恢复工具,支持ext3/ext4双格式分区恢复. 二.安装 yum install -y e2fsprogs* e2fslibs* wget http://nch ...