题意

题目描述

逆时针给出n个凸多边形的顶点坐标,求它们交的面积。例如n=2时,两个凸多边形如下图:

则相交部分的面积为5.233。

输入输出格式

输入格式:

第一行有一个整数n,表示凸多边形的个数,以下依次描述各个多边形。第i个多边形的第一行包含一个整数mi,表示多边形的边数,以下mi行每行两个整数,逆时针给出各个顶点的坐标。

输出格式:

输出文件仅包含一个实数,表示相交部分的面积,保留三位小数。

输入输出样例

输入样例#1:
复制

2
6
-2 0
-1 -2
1 -2
2 0
1 2
-1 2
4
0 -3
1 -1
2 2
-1 0
输出样例#1:
复制

5.233

说明

100%的数据满足:2<=n<=10,3<=mi<=50,每维坐标为[-1000,1000]内的整数

分析

这题数据范围这么小,直接上半平面交模板即可。

点->左闭右开区间,半平面->闭区间。

操作顺序:

  1. 弹双端队列
  2. 加入队列
  3. 判平行
  4. 求交点

最后还要用第一个半平面去除无用的平面。当last-first<=1时,半平面交是空集

时间复杂度\(O(n\log n)\)

代码

注意空集的时候面积输出0仍然要保留三位小数。

算面积的写错了仍然能AC?

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<vector>
#include<cmath>
#define co const
using namespace std;

co double eps=1e-8;
int dcmp(double x) {return fabs(x)<eps?0:(x>0?1:-1);}
typedef struct Point{double x,y;}Vector;
Vector operator+(co Vector&u,co Vector&v) {return (Vector){u.x+v.x,u.y+v.y};}
Vector operator-(co Vector&u,co Vector&v) {return (Vector){u.x-v.x,u.y-v.y};}
Vector operator*(co Vector&u,double k) {return (Vector){u.x*k,u.y*k};}
double cross(co Vector&u,co Vector&v) {return u.x*v.y-u.y*v.x;}
double dot(co Vector&u,co Vector&v) {return u.x*v.x+u.y*v.y;}
struct Line{
    Point p;Vector v;
    double ang;
    bool operator<(co Line&l) {return ang<l.ang;}
};
bool left(co Line&l,co Point&p){
    return dcmp(cross(l.v,p-l.p))>0;
}
Point intersection(co Line&a,co Line&b){
    return a.p+a.v*(cross(b.v,a.p-b.p)/cross(a.v,b.v));
}

int main(){
    int n;scanf("%d",&n);
    vector<Line>ls;
    for(int m;n--;){
        scanf("%d",&m);
        static Point p[50];
        for(int i=0;i<m;++i) scanf("%lf%lf",&p[i].x,&p[i].y);
        for(int i=0;i<m;++i) ls.push_back((Line){p[i],p[(i+1)%m]-p[i]});
    }
    for(int i=0;i<ls.size();++i) ls[i].ang=atan2(ls[i].v.y,ls[i].v.x);
    sort(ls.begin(),ls.end());
    vector<Point> p(ls.size());
    vector<Line> q(ls.size());
    int first,last;
    q[first=last=0]=ls[0];
    for(int i=1;i<ls.size();++i){
        while(first<last&&!left(ls[i],p[last-1])) --last;
        while(first<last&&!left(ls[i],p[first])) ++first;
        q[++last]=ls[i];
        if(dcmp(cross(q[last].v,q[last-1].v))==0){
            --last;
            if(left(q[last],ls[i].p)) q[last]=ls[i];
        }
        if(first<last) p[last-1]=intersection(q[last-1],q[last]);
    }
    while(first<last&&!left(q[first],p[last-1])) --last;
    if(last-first<=1) return puts("0.000"),0;
    p[last]=intersection(q[last],q[first]);
    vector<Point> ans;
    for(int i=first;i<=last;++i) ans.push_back(p[i]);
    double area=0;
    for(int i=1;i<ans.size();++i) area+=cross(ans[i]-ans[0],ans[(i+1)%ans.size()]-ans[0]);
    printf("%.3lf",fabs(area/2));
    return 0;
}

LG4196 [CQOI2006]凸多边形的更多相关文章

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

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

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

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

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

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

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

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

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

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

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

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

  7. bzoj2618: [Cqoi2006]凸多边形

    Description 逆时针给出n个凸多边形的顶点坐标,求它们交的面积.例如n=2时,两个凸多边形如下图: 则相交部分的面积为5.233. Input 第一行有一个整数n,表示凸多边形的个数,以下依 ...

  8. BZOJ2618[Cqoi2006]凸多边形——半平面交

    题目描述 逆时针给出n个凸多边形的顶点坐标,求它们交的面积.例如n=2时,两个凸多边形如下图: 则相交部分的面积为5.233. 输入 第一行有一个整数n,表示凸多边形的个数,以下依次描述各个多边形.第 ...

  9. P4196 [CQOI2006]凸多边形 半平面交

    \(\color{#0066ff}{题目描述}\) 逆时针给出n个凸多边形的顶点坐标,求它们交的面积.例如n=2时,两个凸多边形如下图: 则相交部分的面积为5.233. \(\color{#0066f ...

随机推荐

  1. EF-记录程序自动生成并执行的sql语句日志

    在EntityFramework的CodeFirst模式中,我们想将程序自动生成的sql语句和执行过程记录到日志中,方便以后查看和分析. 在EF的6.x版本中,在DbContext中有一个Databa ...

  2. html回顾随笔JS(*^__^*)

    ---恢复内容开始--- map遍历 function b(){ var week = new Map(); week.set("Mon","星期一"); we ...

  3. <Using parquet with impala>

    Operations upon Impala Create table stored as parquet like parquet '/user/etl/datafile1' stored as p ...

  4. 20165326 java第五周学习笔记

    第五周学习笔记 ch7 内部类(&外嵌类) 内部类的类体不可以声明类变量和方法 内部类如同类的变量或方法 内部类和外嵌类在编译时生成两个class文件 匿名类 某个类的一个子类没有明显的用类声 ...

  5. php usort

    <?phpfunction re($a,$b){ return ($a>$b)?1:-1; }$x=array(1,3,2,5,9);usort($x, 're');print_r($x) ...

  6. Puppet的一些奇技淫巧

    puppet这个工具真的很神奇,先不说商业版有哪些黑科技,单是开源版本就有很多可能让你摸不着头脑的地方,下面来列举一下puppet是怎么查找puppet server的 其实很简单,puppet ag ...

  7. JAVA中的Set

    Set中存放的是没有重复的数据,下说记录一下使用中的小细节. 1.HashSet 区分大小写: Set<String> set1 = new HashSet<String>() ...

  8. 解析XML技术

    转载:http://developer.51cto.com/art/200903/117512.htm XML现在已经成为一种通用的数据交换格式,它的平台无关性,语言无关性,系统无关性,给数据集成与交 ...

  9. ES6 对象的扩展 Object.assign()

    Object.assign方法用于对象的合并,将源对象(source)的所有可枚举属性,复制到目标对象(target). const target = { a: 1 }; const source1 ...

  10. 微软Power BI 每月功能更新系列——11月Power BI 新功能学习

    Power BI Desktop11月产品功能摘要 本月Power BI Desktop 有一个大规模的更新.现在,通常可以使用复合模型在一个模型中将直接查询和导入源组合在一起.UserVoice上的 ...