【BZOJ】1038: [ZJOI2008]瞭望塔
http://www.lydsy.com/JudgeOnline/problem.php?id=1038
题意:给出n个x轴各不相同的二维整点,且升序,n<=300,坐标绝对值<=10^6,求这些点依次连接后折线的上面取一个点(x, y)使得:x0<=x<=xn,且这个点可以看得到所有线段的所有点。要求这些点到(垂直到)折线的y值之差最小。
#include <cstdio>
#include <cstring>
#include <cmath>
#include <string>
#include <iostream>
#include <algorithm>
#include <queue>
#include <set>
#include <map>
#include <sstream>
using namespace std;
typedef long long ll;
#define pb push_back
#define rep(i, n) for(int i=0; i<(n); ++i)
#define for1(i,a,n) for(int i=(a);i<=(n);++i)
#define for2(i,a,n) for(int i=(a);i<(n);++i)
#define for3(i,a,n) for(int i=(a);i>=(n);--i)
#define for4(i,a,n) for(int i=(a);i>(n);--i)
#define CC(i,a) memset(i,a,sizeof(i))
#define read(a) a=getint()
#define print(a) printf("%d", a)
#define dbg(x) cout << (#x) << " = " << (x) << endl
#define error(x) (!(x)?puts("error"):0)
#define rdm(x, i) for(int i=ihead[x]; i; i=e[i].next)
inline int getint() { static int r, k; r=0,k=1; static char c; c=getchar(); for(; c<'0'||c>'9'; c=getchar()) if(c=='-') k=-1; for(; c>='0'&&c<='9'; c=getchar()) r=r*10+c-'0'; return k*r; } const double eps=1e-6, oo=1e60;
const int N=1005;
int dcmp(double x) { return abs(x)<eps?0:(x<0?-1:1); }
struct iP { double x, y; iP(double _x=0, double _y=0) : x(_x), y(_y) {} };
typedef iP iV;
iV operator-(iP a, iP b) { return iV(a.x-b.x, a.y-b.y); }
iP operator+(iP a, iV b) { return iP(a.x+b.x, a.y+b.y); }
iV operator*(iV a, double x) { return iV(a.x*x, a.y*x); }
double cross(iV a, iV b) { return a.x*b.y-a.y*b.x; }
double angle(iV &a) { return atan2(a.y, a.x); } struct iL {
iP p; iV v; double ang;
iL() {}
iL(iP a, iP b) { set(a, b); }
void set(iP a, iP b) { p=a; v=b-a; ang=angle(v); }
bool operator< (const iL &b) const { return ang<b.ang; }
};
iP LLi(iL &a, iL &b) {
static iV u;
static double t;
u=a.p-b.p;
t=cross(b.v, u)/cross(a.v, b.v);
return a.p+a.v*t;
}
bool OnLeft(iP &p, iL &l) { return dcmp(cross(l.v, p-l.p))>0; }
bool half(iL *line, int ln, iP *s, int &cnt) {
static iL a[N], q[N];
static iP b[N];
memcpy(a, line, sizeof(iL)*(ln+1));
sort(a+1, a+1+ln);
int front=0, tail=0;
q[0]=a[1];
for1(i, 2, ln) {
while(front!=tail && !OnLeft(b[tail-1], a[i])) --tail;
while(front!=tail && !OnLeft(b[front], a[i])) ++front;
q[++tail]=a[i];
if(dcmp(cross(q[tail].v, q[tail-1].v))==0) {
--tail;
if(OnLeft(a[i].p, q[tail])) q[tail]=a[i];
}
if(front!=tail) b[tail-1]=LLi(q[tail-1], q[tail]);
}
while(front!=tail && !OnLeft(b[tail-1], q[front])) --tail;
if(tail-front<=1) return 0;
b[tail]=LLi(q[front], q[tail]);
cnt=0;
for1(i, front, tail) s[++cnt]=b[i];
return 1;
} iL line[N];
iP p[N], pp[N], a[N];
int n, ln, num;
void build() {
iP t1, t2;
for1(i, 1, n-1) line[i].set(a[i], a[i+1]);
ln=n-1;
++ln; line[ln].set(iP(oo, oo), iP(-oo, oo));
++ln; line[ln].set(iP(-oo, -oo), iP(oo, -oo));
++ln; line[ln].set(iP(-oo, oo), iP(-oo, -oo));
++ln; line[ln].set(iP(oo, -oo), iP(oo, oo));
}
double cal(iP &a, iP &b, double x) {
double k=(b.y-a.y)/(b.x-a.x), B=a.y-k*a.x;
return k*x+B;
}
int main() {
read(n);
for1(i, 1, n) a[i].x=getint();
for1(i, 1, n) a[i].y=getint();
build();
half(line, ln, pp, num);
double ans=oo;
double mnx=oo, mny=oo; int pos=1;
for1(i, 1, num) if(dcmp(mnx-pp[i].x)>0 || (dcmp(mnx-pp[i].x)==0 && dcmp(mny-pp[i].y)>0)) { mnx=pp[i].x; mny=pp[i].y; pos=i; }
for1(i, 1, num) p[i]=pp[pos], pos=pos%num+1;
int now=2;
for1(i, 1, n) {
while(now<num && dcmp(p[now].x-a[i].x)<0) ++now;
if(dcmp(a[i].x-p[now-1].x)>=0 && dcmp(p[now].x-a[i].x)>=0)
ans=min(ans, cal(p[now], p[now-1], a[i].x)-a[i].y);
}
now=2;
for1(i, 1, num) {
while(now<n && dcmp(a[now].x-p[i].x)<0) ++now;
if(dcmp(p[i].x-a[now-1].x)>=0 && dcmp(a[now].x-p[i].x)>=0)
ans=min(ans, p[i].y-cal(a[now], a[now-1], p[i].x));
}
printf("%.3f\n", ans+eps);
return 0;
}
计算几何题一定要先想好再码!!!而且一定要注意精度问题!!!!因为答案0.00可能变成-0.00.....(一切都是精度的错..),因此我们在输出答案时别忘了加上eps........
本题首先可以发现,可以取的点一定在每条有向线段(就是p[i]到p[i+1])所在直线的左边,所以答案点在这些直线的交中.
即半平面交.....
然后我sb了.............我竟然以为直接二分距离,然后在加一条y=maxh+h的直线然后判断是否存在半平面...........................................显然这样没有考虑到是垂直到达折线段.........................
考虑到,这个半平面交其实是个凸的,而且,他们一定在所有线段的上方...............而每条线段和凸包上的线段都是直的,即线上的点有单调性......因此,最短距离一定在折线端点到另一个图的线段上...
所以分类枚举凸包的点和原来的点判断到另一个凸线的距离即可.......................
求半平面交的算法很简单...........先极角排序,然后用一个双端队列维护,且判断当前直线是否在队列中的直线的交点的左边,如果是则出列.............
然后之前求直线交点喜闻乐见写错一个地方调了1h.............
反正写这题用了3h左右吧......太弱了.............
【BZOJ】1038: [ZJOI2008]瞭望塔的更多相关文章
- bzoj 1038 [ZJOI2008]瞭望塔(半平面交)
[题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=1038 [题意] 找一个最低塔高使可以看到村庄的每一个角落. [思路] 半平面交 能够看 ...
- BZOJ 1038 ZJOI2008 瞭望塔 半平面交
题目大意及模拟退火题解:见 http://blog.csdn.net/popoqqq/article/details/39340759 这次用半平面交写了一遍--求出半平面交之后.枚举原图和半平面交的 ...
- 【BZOJ 1038】 1038: [ZJOI2008]瞭望塔
1038: [ZJOI2008]瞭望塔 Description 致力于建设全国示范和谐小村庄的H村村长dadzhi,决定在村中建立一个瞭望塔,以此加强村中的治安.我们将H村抽象为一维的轮廓.如下图所示 ...
- 1038: [ZJOI2008]瞭望塔 - BZOJ
Description 致力于建设全国示范和谐小村庄的H村村长dadzhi,决定在村中建立一个瞭望塔,以此加强村中的治安.我们将H村抽象为一维的轮廓.如下图所示 我们可以用一条山的上方轮廓折线(x1, ...
- 1038: [ZJOI2008]瞭望塔
半平面交. 半平面指的就是一条直线的左面(也不知道对不对) 半平面交就是指很多半平面的公共部分. 这道题的解一定在各条直线的半平面交中. 而且瞭望塔只可能在各个点或者半平面交折线的拐点处. 求出半平面 ...
- bzoj千题计划126:bzoj1038: [ZJOI2008]瞭望塔
http://www.lydsy.com/JudgeOnline/problem.php?id=1038 本题可以使用三分法 将点按横坐标排好序后 对于任意相邻两个点连成的线段,瞭望塔的高度 是单峰函 ...
- [BZOJ1038][ZJOI2008]瞭望塔(半平面交)
1038: [ZJOI2008]瞭望塔 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 2999 Solved: 1227[Submit][Statu ...
- 【BZOJ1038】[ZJOI2008]瞭望塔 半平面交
[BZOJ1038][ZJOI2008]瞭望塔 Description 致力于建设全国示范和谐小村庄的H村村长dadzhi,决定在村中建立一个瞭望塔,以此加强村中的治安.我们将H村抽象为一维的轮廓.如 ...
- 【BZOJ 1038】[ZJOI2008]瞭望塔
[题目链接]:http://www.lydsy.com/JudgeOnline/problem.php?id=1038 [题意] [题解] 可以看到所有村子的瞭望塔所在的位置只会是在相邻两个村子所代表 ...
随机推荐
- Minimum Size Subarray Sum
Given an array of n positive integers and a positive integer s, find the minimal length of a subarra ...
- 2.10 用最少次数寻找数组中的最大值和最小值[find min max of array]
[本文链接] http://www.cnblogs.com/hellogiser/p/find-min-max-of-array.html [题目] 对于一个由N个整数组成的数组,需要比较多少次才能把 ...
- DP:Wooden Sticks(POJ 1065)
摆木棍 题目大意:即使有一堆木棍,给一个特殊机器加工,木棍都有两个属性,一个是l一个是w,当机器启动的时候(加工第一根木棒的时候),需要一分钟,在这以后,设机器加工的上一根木棒的长度是l,质量是w,下 ...
- BestCoder10 1002 Revenge of GCD(hdu 5019) 解题报告
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5019 题目意思:给出 X 和 Y,求出 第 K 个 X 和 Y 的最大公约数. 例如8 16,它们的公 ...
- poj 1611 The Suspects 解题报告
题目链接:http://poj.org/problem?id=1611 题意:给定n个人和m个群,接下来是m行,每行给出该群内的人数以及这些人所对应的编号.需要统计出跟编号0的人有直接或间接关系的人数 ...
- bootstrap学习总结1
什么是Bootstrap? bootstrap就是一个前端框架,有Twitter公司开发 最大的优点: 开源 : 响应式设计:Bootstrap 的响应式 CSS 能够自适应于台式机.平板电脑和手机. ...
- Hadoop 中文编码相关问题 -- mapreduce程序处理GBK编码数据并输出GBK编码数据(转)
hadoop的hdfs文件系统中,默认的是utf-8, 故你上传的文件是要设置成utf-8.当输入的是gbk,有该如何? 输入是GBK文件, 输出也是 GBK 文件的示例代码: Hadoop处理GBK ...
- Nginx与Redis解决高并发问题
原文链接:http://bbs.phpchina.com/forum.php?mod=viewthread&tid=229629 第一版产品采用的是Jquery,Nginx,PHP(CI框架) ...
- Linux Top
http://www.it165.net/os/html/201402/7262.html
- 滚屏加载--jQuery+PHP实现浏览更多内容
滚屏加载技术,就是使用Javascript监视滚动条的位置,每次当滚动条到达浏览器窗口底部时,触发一个Ajax请求后台PHP程序,返回相应的数据,并将返回的数据追加到页面底部,从而实现了动态加载,其实 ...