hdu5033 Building 单调队列
// hdu5033 Building 单调队列
//
// 题目大意:
//
// n栋大楼,有一个高度h和位置x。如今有一个人高度为0,有q个询问
// 每一个询问有一个位置x,求在位置x能看到天空的最大的角度。
//
// 解题思路:
//
// 首先得想到将q个询问的位置作为一栋大楼放在整个大楼中考虑。这样
// 问题就比較一致,也比較easy处理啦。
//
// 想象一下,每次从左往右走,对于当前位置的左半边的90度内,看天的角度是
// 逐渐减小的,这样。假设当前位置的左边比当前位置要低,那么当前位置就挡住了
// 之前的更矮的。也就是说之后的位置是不可能看到比当前更矮的大楼了,这样。就可
// 以看成是一个单调的序列。序列中元素高度单调递减,即所谓的单调队列。 但仅仅仅仅是
// 这样是不够的。也会存在比当前元素(设为h)高设为h1,在队列中h1前面的元素是h2,尽管
// h1是比h大,可是假设顺着h1的高度看过去。会被h2挡住。这时h1是没有作用的。也要将
// 这个元素从队列中移除。
//
// 当遇到的大楼要查询的位置的时候,此时队列最后的一个元素并不一定是最优的值,由于
// 这种值可能是不合法的。比方上面的h1会被h2挡住。并不会看到天空。 照样移除这种元素
// 最后,队列的最后一个元素就是解。
//
// 右边的问题全然能够转化为左边的问题,仅仅是从右往左处理而已。 //
//
// 感悟:
//
// 这道题是14年北京区的网赛的一道题目,当时的我尽管看懂了题目的意思。可是真的全然不会做。
// 直到如今伟大的MW大咖,说是单调队列能够做,我就做了。可是卡了一天半,还是想不出来详细怎么解
// 首先将询问的位置当作大楼这一点我想到了,维持高度递减的单调队列我想到了,就是最后一种情况没有
// 想到,更不知道要怎么处理,在伟大的MW大咖的敦敦教诲还有耐心的提示下。最终最终想到了解决的办法
// 不easy啊不easy。在此真诚感谢MW大咖~
//
// 过程无疑会有非常多的疑惑,一个地方不慎。满盘皆输,找到一处的错误。心里十分的欣喜,特别是在ac
// 之后。那种心情。实在是难以描写叙述。感觉到自己的付出,真的是有回报。曾经不懂的东西,自己认真学
// 总会有收获。哪怕收获是那么一点点,微不足道,但收获就是收获,没有这个能让人更加欣喜啦,痛苦并快乐着
// 十分的享受。还是那句话,继续练吧~~~ #include <cstdio>
#include <algorithm>
#include <cmath>
#include <iostream>
#include <cstring>
using namespace std; const int MAX_N = 2e5 + 9; const double PI = acos(-1.0);
int n,q;
double deqh[MAX_N];
double deqx[MAX_N];
double angle[MAX_N]; struct node {
double pos;
double h;
int id; node(){ } node(double pos,double h,int id):pos(pos),h(h),id(id){ }
}; node sky[MAX_N]; bool cmp(node a,node b){
return a.pos < b.pos;
} void input(){
scanf("%d",&n);
double x,h;
for (int i = 1;i <= n;i++){
scanf("%lf%lf",&x,&h);
sky[i] = node(x,h,0);
}
scanf("%d",&q); for (int i = 1;i <= q;i++){
scanf("%lf",&x);
sky[i+n] = node(x,0.0,i);
angle[i] = 0.0;
} n += q; sort(sky+1,sky+n+1,cmp);
} double getk(double a,double b){
return fabs(a/b);
} void getleft(){
int head,tail;
head = tail = 0;
for (int i=1;i<=n;i++){
if (sky[i].id){
while(head + 1 < tail){
double k1 = getk(deqh[tail-1],deqx[tail-1]-sky[i].pos);
double k2 = getk(deqh[tail-2],deqx[tail-2]-sky[i].pos);
if (k1 <= k2)
tail--;
else break;
}
angle[sky[i].id] += atan(deqh[tail-1]/fabs(deqx[tail-1]-sky[i].pos));
}else { while(head < tail && deqh[tail-1] < sky[i].h)
tail--; while(head + 1 < tail){
double k1 = getk(deqh[tail-1]-sky[i].h,deqx[tail-1]-sky[i].pos);
double k2 = getk(deqh[tail-2]-sky[i].h,deqx[tail-2]-sky[i].pos);
if (k1 <= k2)
tail--;
else
break;
}
deqh[tail] = sky[i].h;
deqx[tail++] = sky[i].pos;
}
}
} void getright(){
int head,tail;
head = tail = 0;
for (int i = n;i >= 1;i--){
if (sky[i].id){
while(head + 1 < tail){
double k1 = getk(deqh[tail-1],deqx[tail-1]-sky[i].pos);
double k2 = getk(deqh[tail-2],deqx[tail-2]-sky[i].pos);
if (k1 <= k2)
tail--;
else
break;
} angle[sky[i].id] += atan(deqh[tail-1]/fabs(deqx[tail-1]-sky[i].pos)); }else {
while(head < tail && deqh[tail-1] < sky[i].h)
tail--; while(head + 1 < tail){ double k1 = getk(deqh[tail-1]-sky[i].h,deqx[tail-1] - sky[i].pos);
double k2 = getk(deqh[tail-2]-sky[i].h,deqx[tail-2] - sky[i].pos); if (k1 <= k2){
tail--;
}else
break; }
deqh[tail] = sky[i].h;
deqx[tail++] = sky[i].pos;
}
}
} void solve(){
getleft();
getright();
for (int i=1;i<=q;i++){
printf("%.10lf\n",(PI - angle[i]) * 180.0 / PI);
}
} int main(){
int t;
//freopen("1.txt","r",stdin);
scanf("%d",&t);
int kase = 1;
while(t--){
input();
printf("Case #%d:\n",kase++);
solve();
}
}
hdu5033 Building 单调队列的更多相关文章
- HDU5033 building 单调栈+计算几何
正解:单调栈 解题报告: 哇生气辽QAQ本来打了半天feel good都快调出来了然后说换题了QAQ(所以可能那题的代码会过一阵子再放上来了QAQ 不过还是大爆手速打了一通拿到首杀了嘻嘻 美滋滋辽 然 ...
- HDU5033 Building(单调栈)
题意是说在水平轴上有很多建筑物(没有宽度),知道每个建筑物的位置与高度.有m个查询,每次查询位置x所能看到的天空的角度. 方法是将建筑与查询一起排序,从左往右计算一遍,如果是建筑物,则比较最后两个(当 ...
- hdu5033 Building (单调栈+)
http://acm.hdu.edu.cn/showproblem.php?pid=5033 2014 ACM/ICPC Asia Regional Beijing Online B 1002 Bui ...
- [hdu5033]单调队列
题意:x轴上有n棵树,询问你站在某个点的视角.从左至右,单调队列(类似凸包)维护下.我强迫症地写了个模板QAQ #include <iostream> #include <cstdi ...
- BestCoder Round #89 B题---Fxx and game(单调队列)
题目链接 http://acm.hdu.edu.cn/showproblem.php?pid=5945 问题描述 输入描述 输出描述 输入样例 输出样例 题意:中文题,不再赘述: 思路: B ...
- 单调队列 && 斜率优化dp 专题
首先得讲一下单调队列,顾名思义,单调队列就是队列中的每个元素具有单调性,如果是单调递增队列,那么每个元素都是单调递增的,反正,亦然. 那么如何对单调队列进行操作呢? 是这样的:对于单调队列而言,队首和 ...
- FZU 1914 单调队列
题目链接:http://acm.fzu.edu.cn/problem.php?pid=1914 题意: 给出一个数列,如果它的前i(1<=i<=n)项和都是正的,那么这个数列是正的,问这个 ...
- BZOJ 1047 二维单调队列
题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=1047 题意:见中文题面 思路:该题是求二维的子矩阵的最大值与最小值的差值尽量小.所以可以考 ...
- 【BZOJ3314】 [Usaco2013 Nov]Crowded Cows 单调队列
第一次写单调队列太垃圾... 左右各扫一遍即可. #include <iostream> #include <cstdio> #include <cstring> ...
随机推荐
- (十一)Unity5新特性----实战2D游戏
孙广东 2015.7.11 在本教程中,将了解到U5新功能.你通过本教程.您将了解下面内容: Changes in Component Access Physics Effectors Adding ...
- 遗传奥秘的伟大揭秘者:J.Watson
J.Watson的近照: 人们公认,揭秘生命体的遗传奥秘(DNA)是二十世纪最伟大的科技成果之中的一个,或许就是人类最伟大的科技进步(而不是"之中的一个"). 上世纪是人类做出伟大 ...
- UVA - 1642 Magical GCD 数学
Magical GCD The Magical GCD of a nonempty sequence of positive integer ...
- 一个JavaWeb项目中使用的部分技术
-- 2015年8月8日 1. Web框架: Spring+ SpringMVC + MyBatis Spring: 作为容器.工厂,用于解耦以及管理对象生命周期. 整合各类框架和依赖. MVC : ...
- 机器学习 数据量不足问题----1 做好特征工程 2 不要用太多的特征 3 做好交叉验证 使用线性svm
来自:https://www.zhihu.com/question/35649122 其实这里所说的数据量不足,可以换一种方式去理解:在维度高的情况下,数据相对少.举一个特例,比如只有一维,和1万个数 ...
- 7个好用的在线YouTube视频下载工具
title: 7个好用的在线YouTube视频下载工具 toc: false date: 2018-10-10 15:11:00 categories: methods tags: youtube C ...
- 弹出ifame页面(jquery.reveal.js)
<body> <a data-reveal-id="myModalDailyModify" data-animation="fade" tit ...
- Python 递归和二分查找
# 二分查找l1 = [2,3,5,10,15,16,18,22,26,30,32,35,41,42,43,55,56,66,67,69,72,76,82,83,88] def two_search( ...
- TFS源代码管理工具:
源代码管理: 先获取最新版本,再签入.如发现错误,可以点击--源代码管理--获取特定版本撤回修改 1.签入:(要备注,测试通过后签入) 敏捷开发:(小步快跑):小部分功能开发完成测试通过后就签入 全部 ...
- DOM基础知识(Node对象、Element对象)
5.Node对象 u 遍历节点 u 父节点 .parentNode - 获取父节点—> 元素节点或文档节点 .parentElement - 获取父元素节点—> 元素节点 u 子节 ...