很明显是一道半平面交的题。

先说一下半平面交的步骤:

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]凸多边形(半平面交)的更多相关文章

  1. bzoj 2618: [Cqoi2006]凸多边形 [半平面交]

    2618: [Cqoi2006]凸多边形 半平面交 注意一开始多边形边界不要太大... #include <iostream> #include <cstdio> #inclu ...

  2. 洛谷 P4196 [CQOI2006]凸多边形 (半平面交)

    题目链接:P4196 [CQOI2006]凸多边形 题意 给定 \(n\) 个凸多边形,求它们相交的面积. 思路 半平面交 半平面交的模板题. 代码 #include <bits/stdc++. ...

  3. BZOJ - 2618 凸多边形 (半平面交)

    题意:求n个凸多边形的交面积. 半平面交模板题. #include<bits/stdc++.h> using namespace std; typedef long long ll; ty ...

  4. 【BZOJ 2618】 2618: [Cqoi2006]凸多边形 (半平面交)

    2618: [Cqoi2006]凸多边形 Description 逆时针给出n个凸多边形的顶点坐标,求它们交的面积.例如n=2时,两个凸多边形如下图: 则相交部分的面积为5.233. Input 第一 ...

  5. bzoj 2618 2618: [Cqoi2006]凸多边形(半平面交)

    2618: [Cqoi2006]凸多边形 Time Limit: 5 Sec  Memory Limit: 128 MBSubmit: 656  Solved: 340[Submit][Status] ...

  6. 【BZOJ2618】[CQOI2006]凸多边形(半平面交)

    [BZOJ2618][CQOI2006]凸多边形(半平面交) 题面 BZOJ 洛谷 题解 这个东西就是要求凸多边形的边所形成的半平面交. 那么就是一个半平面交模板题了. 这里写的是平方的做法. #in ...

  7. 2018.07.04 BZOJ 2618 Cqoi2006凸多边形(半平面交)

    2618: [Cqoi2006]凸多边形 Time Limit: 5 Sec Memory Limit: 128 MB Description 逆时针给出n个凸多边形的顶点坐标,求它们交的面积.例如n ...

  8. 【BZOJ-2618】凸多边形 计算几何 + 半平面交 + 增量法 + 三角剖分

    2618: [Cqoi2006]凸多边形 Time Limit: 5 Sec  Memory Limit: 128 MBSubmit: 959  Solved: 489[Submit][Status] ...

  9. 算法复习——半平面交(bzoj2618凸多边形)

    讲解: 这里套用wuvin神犇的ppt,附上友情链接:http://blog.leanote.com/wuvin 半平面交: 算法流程: 注意事项: 例题: Description 逆时针给出n个凸多 ...

随机推荐

  1. 「小程序JAVA实战」 小程序私有页面的生命周期以及导航(10)

    转自:https://idig8.com/2018/08/09/xiaochengxu-chuji-10/ 之前讲了小程序全局的生命周期,今天咱们说说单个页面的生命周期!源码:https://gith ...

  2. Delphi IOS 蓝牙锁屏后台运行

    Delphi IOS 后台运行 同样的程序,编译成android,锁屏后继续运行正常,蓝牙通讯正常,但在IOS下锁屏后程序的蓝牙就中断通讯了? IOS的机制就是这样,锁屏就关闭了. 音乐播放器是怎么做 ...

  3. AOP操作术语

  4. realsense and Mask_RCNN

    ###################librealsense and Mask_RCNN cd RealSennse/librealsense2018091501/librealsense/wrap ...

  5. 633. Sum of Square Numbers 是否由两个完全平方数构成

    [抄题]: Given a non-negative integer c, your task is to decide whether there're two integers a and b s ...

  6. 关于sleep的理解

    unix是按时间片轮转调度, windows是抢占式调度 以吃蛋糕为例子,10个人吃蛋糕,如果是unix下, 假设开始时,每个人都处于就绪状态,那么操作系统调度大家排好队,按顺序吃,每个人吃1分钟, ...

  7. Servlet和JSP简述

    什么是Servlet和JSP 用Java开发Web应用程序时用到的技术主要有两种,即Servlet和JSP. Servlet是在服务器端执行的Java程序,一个被称为Servlet容器的程序(其实就是 ...

  8. UOJ#46. 【清华集训2014】玄学

    传送门 分析 清华集训真的不是人做的啊嘤嘤嘤 我们可以考虑按操作时间把每个操作存进线段树里 如果现在点x正好使一个整块区间的右端点则更新代表这个区间的点 我们不难发现一个区间会因为不同的操作被分成若干 ...

  9. Luogu 3350 [ZJOI2016]旅行者

    BZOJ 4456 听若干个大佬讲过$n$遍终于写掉了. 我把时限基本上跑满了2333…… 分治 + 最短路. 首先我们去分治这个矩形格子,找到一条长边把它对半切,对切开的边上的每一个点跑一遍最短路然 ...

  10. csv、txt读写及模式介绍

    1读写模式 r以读方式打开文件,可读取文件信息 w已写方式打开文件,可向文件写入信息.如文件存在,则清空,再写入 a以追加模式打开文件,打开文件可指针移至末尾,文件不存在则创建 r+以读写方式打开文件 ...