BZOJ1845 [Cqoi2005] 三角形面积并 扫描线 计算几何
欢迎访问~原文出处——博客园-zhouzhendong
去博客园看该题解
题目传送门 - BZOJ1845
题意概括
给出n个三角形,求其面积并。
题解
有一个很经典的扫描线题目:矩形面积并。那个比较简单,建议先去看看 —— 传送门 - 矩形面积并。
这个扫描线的算法,我之前就看过。
之前想了想,还以为是n4logn的,自己以为理解错了,所以就弃坑了一段时间。
现在再想想,原来之前思考的是对的,只是复杂度想错了。其实是n3logn的。
我们按照Y来排序,同样的,我们来看一组图片。
然后大概你已经深有感触了。
步骤:把所有的分成多个梯形,然后扫描线解决。
在这之前,我们需要把所有的交点都求出来。
当然,对于上例第一层和第二层的情况十分坑。第二条扫描线的有效长度不仅和其本身的参数有关,还和该边放在上面还是下面有关。
最后,诡异的是,ans要减去Eps才可以A掉。当然也有大佬不减的Orz,题目卡精度啊!
做个好人,代码后面一组数据。
代码
#include <cstring>
#include <algorithm>
#include <cstdio>
#include <cstdlib>
#include <cmath>
using namespace std;
typedef long double LD;
const int N=300+5;
const LD Eps=1e-8,Inf=1e9;
int Dcmp(LD x){
if (fabs(x)<Eps)
return 0;
return x<0?-1:1;
}
struct Point{
LD x,y;
Point (){}
Point (LD x_,LD y_){
x=x_,y=y_;
}
Point operator + (Point a){
return Point(x+a.x,y+a.y);
}
Point operator - (Point a){
return Point(x-a.x,y-a.y);
}
Point operator * (LD a){
return Point(x*a,y*a);
}
Point operator / (LD a){
return Point(x/a,y/a);
}
bool operator == (Point a){
return !Dcmp(x-a.x)&&!Dcmp(y-a.y);
}
void read(){
scanf("%Lf%Lf",&x,&y);
}
};
LD Dot(Point a,Point b){
return a.x*b.x+a.y*b.y;
}
LD Cross(Point a,Point b){
return a.x*b.y-a.y*b.x;
}
LD Cross(Point a,Point b,Point c){
return Cross(b-a,c-a);
}
struct Line{
Point a,b;
Line (){}
Line (Point a_,Point b_){
a=a_,b=b_;
}
Line Mat(){
return Line(Point(min(a.x,b.x),min(a.y,b.y)),Point(max(a.x,b.x),max(a.y,b.y)));
}
};
LD Length(Line a){
return sqrt(Dot(a.a-a.b,a.a-a.b));
}
Point Cut_Point(Line L,LD y){
Point a=L.a,b=L.b;
LD da=y-a.y,dy=b.y-a.y,dx=b.x-a.x;
if (!Dcmp(dy))
return Point(Inf,Inf);
return Point(a.x+dx/dy*da,y);
}
Line OrderedY_Line(Point a,Point b){
if (Dcmp(a.x-b.x)>0)
swap(a,b);
return Line(a,b);
}
bool cmp_x(Point a,Point b){
return Dcmp(a.x-b.x)<0;
}
bool cmp_y(Point a,Point b){
return Dcmp(a.y-b.y)<0;
}
struct Triangle{
Point P[3];
Line L[3];
void build(Point x,Point y,Point z){
P[0]=x,P[1]=y,P[2]=z;
L[0]=Line(x,y),L[1]=Line(x,z),L[2]=Line(y,z);
}
Line Cut(int type,LD Y,bool &flag){
flag=1;
Point P_[3];
for (int i=0;i<3;i++)
P_[i]=P[i];
sort(P_,P_+3,cmp_y);
if (Dcmp(P_[1].y-Y)==0&&Dcmp(P_[2].y-Y)==0)
swap(P_[0],P_[2]);
if (Dcmp(P_[0].y-Y)==0&&Dcmp(P_[1].y-Y)==0){
if (type==0){
if (Dcmp(P_[2].y-Y)<0){
flag=0;
return Line(Point(Inf,Inf),Point(Inf,Inf));
}
return OrderedY_Line(P_[0],P_[1]);
}
else {
if (Dcmp(P_[2].y-Y)>0){
flag=0;
return Line(Point(Inf,Inf),Point(Inf,Inf));
}
return OrderedY_Line(P_[0],P_[1]);
}
}
Point p[3];
Line M[3];
for (int i=0;i<3;i++){
p[i]=Cut_Point(L[i],Y);
M[i]=L[i].Mat();
}
if (Dcmp(Y-min(M[0].a.y,M[1].a.y))<0||Dcmp(Y-max(M[0].b.y,M[1].b.y))>0){
flag=0;
return Line(Point(Inf,Inf),Point(Inf,Inf));
}
for (int i=0;i<3;i++)
if (Dcmp(Y-M[i].a.y)<0||Dcmp(Y-M[i].b.y)>0)
return OrderedY_Line(p[(i+1)%3],p[(i+2)%3]);
sort(p,p+3,cmp_x);
return OrderedY_Line(p[0],p[0]==p[1]?p[2]:p[1]);
}
};
bool Crossed(Line a,Line b){
return Dcmp(Cross(a.a,a.b,b.a))*Dcmp(Cross(a.a,a.b,b.b))<0&&Dcmp(Cross(b.a,b.b,a.a))*Dcmp(Cross(b.a,b.b,a.b))<0;
}
Point Cross_Point(Line a,Line b){
Point P=a.a,Q=b.a,v=a.b-a.a,w=b.b-b.a,u=P-Q;
LD t=Cross(w,u)/Cross(v,w);
return P+v*t;
}
int n,m;
Triangle Tri[N];
LD Y[N*N];
bool cmp_Lx(Line a,Line b){
return Dcmp(a.a.x-b.a.x)<0;
}
LD GetLen(int type,int pos_Y){
LD y=Y[pos_Y];
Line L[N];
int tot=0;
for (int i=1;i<=n;i++){
bool flag;
L[++tot]=Tri[i].Cut(type,y,flag);
if (!flag)
tot--;
}
if (tot==0)
return 0;
sort(L+1,L+tot+1,cmp_Lx);
LD ans=L[1].b.x-L[1].a.x,max_x=L[1].b.x;
for (int i=2;i<=tot;i++){
max_x=max(max_x,L[i].a.x);
if (Dcmp(max_x-L[i].b.x)>0)
continue;
ans+=L[i].b.x-max_x;
max_x=max(max_x,L[i].b.x);
}
return ans;
}
int main(){
scanf("%d",&n);
for (int i=1;i<=n;i++){
Point x,y,z;
x.read(),y.read(),z.read();
Tri[i].build(x,y,z);
}
m=0;
for (int i=1;i<=n;i++)
for (int j=i+1;j<=n;j++)
for (int u=0;u<3;u++)
for (int v=0;v<3;v++)
if (Crossed(Tri[i].L[u],Tri[j].L[v]))
Y[++m]=Cross_Point(Tri[i].L[u],Tri[j].L[v]).y;
for (int i=1;i<=n;i++)
for (int j=0;j<3;j++)
Y[++m]=Tri[i].P[j].y;
sort(Y+1,Y+m+1);
LD ans=0,pre=GetLen(0,1);
for (int i=2;i<=m;i++){
if (Dcmp(Y[i]-Y[i-1])==0)
continue;
ans+=0.5*(pre+GetLen(1,i))*(Y[i]-Y[i-1]);
pre=GetLen(0,i);
}
printf("%.2Lf",ans-Eps);
return 0;
}
数据
2
0 0 2 0 0 3
2 0 0 2 3 2
ans=5.33
数据解释:
BZOJ1845 [Cqoi2005] 三角形面积并 扫描线 计算几何的更多相关文章
- bzoj 1845: [Cqoi2005] 三角形面积并 扫描线
1845: [Cqoi2005] 三角形面积并 Time Limit: 3 Sec Memory Limit: 64 MBSubmit: 848 Solved: 206[Submit][Statu ...
- BZOJ1845 : [Cqoi2005] 三角形面积并
求出所有交点后从左往右扫描线,用每段的中位线去截所有三角形,算出长度并后乘以该段长度即可,时间复杂度$O(n^3\log n)$. #include<cstdio> #include< ...
- BZOJ 1845: [Cqoi2005] 三角形面积并 [计算几何 扫描线]
1845: [Cqoi2005] 三角形面积并 Time Limit: 3 Sec Memory Limit: 64 MBSubmit: 1151 Solved: 313[Submit][Stat ...
- 【BZOJ1845】[Cqoi2005] 三角形面积并 几何+扫描线
[BZOJ1845][Cqoi2005] 三角形面积并 Description 给出n个三角形,求它们并的面积. Input 第一行为n(N < = 100), 即三角形的个数 以下n行,每行6 ...
- CQOI2005 三角形面积并 和 POJ1177 Picture
1845: [Cqoi2005] 三角形面积并 Time Limit: 3 Sec Memory Limit: 64 MBSubmit: 1664 Solved: 443[Submit][Stat ...
- [CQOI2005]三角形面积并
[CQOI2005]三角形面积并 题目大意: 求\(n(n\le100)\)个三角形的面积并. 思路: 自适应辛普森法,玄学卡精度可过. 源代码: #include<cmath> #inc ...
- BZOJ 1845: [Cqoi2005] 三角形面积并 (辛普森积分)
大力辛普森积分 精度什么的搞了我好久- 学到了Simpson的一个trick 深度开11,eps开1e-4.跑的比有些扫描线还快- CODE #include <bits/stdc++.h> ...
- UVa 11437:Triangle Fun(计算几何综合应用,求直线交点,向量运算,求三角形面积)
Problem ATriangle Fun Input: Standard Input Output: Standard Output In the picture below you can see ...
- hdu 4709:Herding(叉积求三角形面积+枚举)
Herding Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Sub ...
随机推荐
- CentOS6.8配置SonarQube Scanner配合SonarQube使用
下载最新的SonarQube Scanner压缩包 https://docs.sonarqube.org/display/SCAN/Analyzing+with+SonarQube+Scanner ...
- mysql一次查询,返回多个统计结果
1.sum(if) select sum(if(status=1,1,0)) as s1_count,sum(if(status=2,1,0)) as s2_countfrom order; 2.co ...
- Linux CentOS 服务器搭建与初始化配置图文详解
这几天对服务器兴趣贼为浓厚,在虚拟机上装了一个CentOS7玩了玩,遇到过很多问题,比如网卡驱动设置,不能ping 等等问题,然后掏钱买个ECS搭服务器玩玩,下面就开始谢谢我的心路历程吧. 首先 买服 ...
- java工程师之旅-一个月工作心得
不知不觉,在工作中已经度过一个月,距离上次写文章已经好几个月了,正好还有二十分钟下班,抽点时间来写一下博文,写一下心得. 首先说一下,在我工作之前,做了一个项目,和一个外校大四的学生做一个毕业设计,一 ...
- 解决ping 127.0.0.1 一般故障 问题
故障如下图: 绕了好一大圈才发现是goupi防火墙搞的鬼,弄得我一些软件一直运行不了!!!!! 废话不多说,关了防火墙就行了:操作步骤如下图示 关闭之后,美滋滋:
- 使用密钥认证机制远程登录Linux
密钥认证机制 创建存放key的文件 1)创建目录 /root/.ssh 并设置权限 [root@localhost ~]# mkdir /root/.ssh mkdir 命令用来创建目录,以后会详细介 ...
- php数据库的增删改查
1.查询: 数据的显示,这里就可以嵌入php来进行数据的输出 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 ...
- WebRTC服务器——Licode 环境搭建
WebRTC服务器--Licode 环境搭建 系统配置 阿里云服务器 Ubuntu 14.04.5 LTS Docker 环境搭建 在一台空的机器上搭建docker环境,先要安装docker,执行下面 ...
- openvpn用户管理、linux客户端配置及企业常用真实案例解析
1.给企业用户分配VPN账户的流程: 添加拨号需要密码的用户 # source vars NOTE: If you run ./clean-all, I will be doing a rm -rf ...
- IE中window的模态框与返回值
window.returnValue是javascript中html的window对象的属性,目的是返回窗口值,当用window.showModalDialog函数打开一个IE的模态窗口时,用于返回窗 ...