BZOJ1038 瞭望塔
学习了半平交面。
我这里写的是训练指南中的双端队列,每次判断是否删去更优然后更新。
看hzwer中有一处不太明白就是为何要将两段加入队列
后来对拍出错才知道是因为精度,当两线重合时他们叉积返回值是一个极小值
所以判断一下精度即可。
#include<bits/stdc++.h>
using namespace std;
const int N=;
int n,m,cnt,tot;double ans=1e20,eps=1e-;
struct point{
double x,y;
}a[N],p[N];
point operator -(point a,point b){
point t;t.x=b.x-a.x;t.y=b.y-a.y;
return t;
}
double operator *(point a,point b){
return a.x*b.y-a.y*b.x;
}
struct line{
point a,b;double slop;
bool operator <(const line &B)const{
return slop==B.slop?(b-a)*(B.b-a)>:slop<B.slop;
}
}l[N],s[N];
point inter(line a,line b){
double k1,k2,t;
k1=(b.b-a.a)*(a.b-a.a);
k2=(a.b-a.a)*(b.a-a.a);
t=k1/(k1+k2);
point ans;
ans.x=b.b.x+(b.a.x-b.b.x)*t;
ans.y=b.b.y+(b.a.y-b.b.y)*t;
return ans;
}
bool jud(line a,line b,line t)
{
point pp=inter(a,b);
return (t.b-t.a)*(pp-t.a)<-eps;
}
void hpl()
{
int L=,R=;
for(int i=;i<=cnt;++i)
{
if(l[i].slop!=l[i-].slop)++tot;
l[tot]=l[i];
}
cnt=tot;tot=;
s[++R]=l[];s[++R]=l[];
for(int i=;i<=cnt;++i)
{
while(L<R&&jud(s[R-],s[R],l[i]))R--;
while(L<R&&jud(s[L+],s[L],l[i]))L++;
s[++R]=l[i];
}
while(L<R&&jud(s[R-],s[R],s[L]))R--; while(L<R&&jud(s[L+],s[L],s[R]))L++;
for(int i=L;i<R;++i)
a[++tot]=inter(s[i],s[i+]);
}
void getans()
{
for(int i=;i<=tot;++i)
{
point t;t.x=a[i].x;t.y=-;
for(int j=;j<n;++j)
if(p[j].x<=a[i].x&&p[j+].x>=a[i].x)
ans=min(ans,a[i].y-inter((line){p[j],p[j+]},(line){t,a[i]}).y);
}
for(int i=;i<=n;++i)
{
point t;t.x=p[i].x;t.y=-;
for(int j=;j<tot;++j)
if(a[j].x<=p[i].x&&a[j+].x>=p[i].x)
ans=min(ans,inter((line){a[j],a[j+]},(line){t,p[i]}).y-p[i].y);
}
}
int main()
{
// freopen("1.out","r",stdin);
// freopen("my.out","w",stdout);
scanf("%d",&n);
for(int i=;i<=n;++i)scanf("%lf",&p[i].x);
for(int i=;i<=n;++i)scanf("%lf",&p[i].y);
// p[0].x=p[1].x;p[0].y=1e9;
// p[n+1].x=p[n].x;p[n+1].y=1e9;
for(int i=;i<n;++i)l[++cnt].a=p[i],l[cnt].b=p[i+];
if(cnt<){
puts("0.000");return ;
}
for(int i=;i<=cnt;++i)
l[i].slop=atan2(l[i].b.y-l[i].a.y,l[i].b.x-l[i].a.x);
sort(l+,l++cnt);
hpl();
getans();
printf("%.3lf",ans);
return ;
}
BZOJ1038 瞭望塔的更多相关文章
- 【bzoj1038】瞭望塔
[bzoj1038]瞭望塔 题意 致力于建设全国示范和谐小村庄的H村村长dadzhi,决定在村中建立一个瞭望塔,以此加强村中的治安.我们将H村抽象为一维的轮廓.如下图所示 我们可以用一条山的上方轮廓折 ...
- 【BZOJ1038】[ZJOI2008]瞭望塔 半平面交
[BZOJ1038][ZJOI2008]瞭望塔 Description 致力于建设全国示范和谐小村庄的H村村长dadzhi,决定在村中建立一个瞭望塔,以此加强村中的治安.我们将H村抽象为一维的轮廓.如 ...
- bzoj1038: [ZJOI2008]瞭望塔
Description 致力于建设全国示范和谐小村庄的H村村长dadzhi,决定在村中建立一个瞭望塔,以此加强村中的治安.我们将H村抽象为一维的轮廓.如下图所示 我们可以用一条山的上方轮廓折线(x1, ...
- bzoj千题计划126:bzoj1038: [ZJOI2008]瞭望塔
http://www.lydsy.com/JudgeOnline/problem.php?id=1038 本题可以使用三分法 将点按横坐标排好序后 对于任意相邻两个点连成的线段,瞭望塔的高度 是单峰函 ...
- 【bzoj1038】瞭望塔 半平面交
题目描述 致力于建设全国示范和谐小村庄的H村村长dadzhi,决定在村中建立一个瞭望塔,以此加强村中的治安.我们将H村抽象为一维的轮廓.如下图所示 我们可以用一条山的上方轮廓折线(x1, y1), ( ...
- 【BZOJ1038】【ZJOI2008】瞭望塔 [模拟退火]
瞭望塔 Time Limit: 10 Sec Memory Limit: 162 MB[Submit][Status][Discuss] Description 致力于建设全国示范和谐小村庄的H村村 ...
- 「BZOJ1038」「洛谷P2600」「ZJOI2008」瞭望塔 半平面交+贪心
题目链接 BZOJ/洛谷 题目描述 致力于建设全国示范和谐小村庄的H村村长dadzhi,决定在村中建立一个瞭望塔,以此加强村中的治安. 我们将H村抽象为一维的轮廓.如下图所示: 我们可以用一条山的上方 ...
- [BZOJ1038][ZJOI2008]瞭望塔(半平面交)
1038: [ZJOI2008]瞭望塔 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 2999 Solved: 1227[Submit][Statu ...
- 刷题总结——瞭望塔(bzoj1038)
题目: Description 致力于建设全国示范和谐小村庄的H村村长dadzhi,决定在村中建立一个瞭望塔,以此加强村中的治安.我们将H村抽象为一维的轮廓.如下图所示 我们可以用一条山的上方轮廓折线 ...
随机推荐
- Lua的各种资源2
Lua Directory This page is a top level directory of all Lua content at this wiki, grouped by top ...
- layer弹窗的跳转功能
1,本弹窗直接跳转父页面: @if(session('message')) <script> window.parent.location.reload(); //刷新父页面 var in ...
- koa源码阅读[1]-koa与koa-compose
接上次挖的坑,对koa2.x相关的源码进行分析 第一篇.不得不说,koa是一个很轻量.很优雅的http框架,尤其是在2.x以后移除了co的引入,使其代码变得更为清晰. express和koa同为一批人 ...
- 蓝色的PC端后台管理界面设计模板——后台
链接:http://pan.baidu.com/s/1o82hXX4 密码:x6le
- python基础===15条变量&方法命名的最佳实践
不同的代码段采用不同的命名长度.通常来说,循环计数器(loop counters)采用1位的单字符来命名,循环判断变量(condition/loop variables)采用1个单词来命名,方法采用1 ...
- MySQL登录问题1045 (28000)处理步骤【原创】
MySQL登录问题1045 (28000) 俩台服务器主从复制,从的同步账号无法远程登录主服务器.报错ERROR 1045 (28000): Access denied for user 'root ...
- Redis使用详细教程【转】
转自 Redis使用详细教程 - wangyuyu - 博客园http://www.cnblogs.com/wangyuyu/p/3786236.html 一.Redis基础部分: 1.redis介绍 ...
- rust 入门
hello rust fn main() { println!("Hello, world!"); } 从hello world入手,rust的语法是比较简洁. 在mac os中, ...
- 函数参数 f_arg, *args, **kwargs
当需要给函数传参时,可以通过运用不同的形参来传递,达到参数不同的使用目的. 简单来说:f_arg就是传递的第一个参数,类似于C++中的普通参数: *args 传递的是一个参数的list: **kwar ...
- Dubbo使用
[注:本文参考<Dubbo入门---搭建一个最简单的Demo框架>,感谢原创作者的知识探索与奉献] 一.Dubbo背景和简介 Dubbo开始于电商系统,因此在这里先从电商系统的演变讲起. ...