[Luogu P4198]楼房重建(线段树)
题目描述
小A的楼房外有一大片施工工地,工地上有N栋待建的楼房。每天,这片工地上的房子拆了又建、建了又拆。他经常无聊地看着窗外发呆,数自己能够看到多少栋房子。
为了简化问题,我们考虑这些事件发生在一个二维平面上。小A在平面上(0,0)点的位置,第i栋楼房可以用一条连接(i,0)和(i,Hi)的线段 表示,其中Hi为第i栋楼房的高度。如果这栋楼房上任何一个高度大于0的点与(0,0)的连线没有与之前的线段相交,那么这栋楼房就被认为是可见的。
施工队的建造总共进行了M天。初始时,所有楼房都还没有开始建造,它们的高度均为0。在第i天,建筑队将会将横坐标为Xi的房屋的高度变为Yi(高 度可以比原来大—修建,也可以比原来小—拆除,甚至可以保持不变—建筑队这天什么事也没做)。请你帮小A数数每天在建筑队完工之后,他能看到多少栋楼房?
输入输出格式
输入格式:
第一行两个正整数N,M
接下来M行,每行两个正整数Xi,Yi
输出格式:
M行,第i行一个整数表示第i天过后小A能看到的楼房有多少栋
输入输出样例
说明
对于所有的数据1<=Xi<=N,1<=Yi<=10^9
N,M<=100000
线段树博大精深。
y/x作为每个点的值,求最靠左的极长上升子序列,线段树维护。
考虑如何合并两个区间,设c[x]为x区间内的子序列长度。显然是左边的极长上升子序列选完,然后右边选出第一个数大于max{左边所有数(也就是左边子序列末尾)}的极长上升子序列。
至于怎么求一个区间内第一个数大于某个值的子序列长度,再用一个函数解决,que(x,k)表示x节点的区间内大于k的极长上升子序列的长度,递归查询即可。
实现起来会发现是c[]和find()的相互利用,比较巧妙。
因为修改会带上查询,所以复杂度$O(n\log^2 n)$
另外这题数据类型是实数,会发现实数比较的时候如果完全不考虑精度问题(直接用>=之类的运算)的话不存在问题,而如果考虑了之后eps设宽了反而拿不到分,eps设到1e-10才能拿到分。
仿佛又看到了NOIP2017D2T1的惨败。
#include<cmath>
#include<cstdio>
#include<algorithm>
#define ls (x<<1)
#define rs (ls|1)
using namespace std; const int N=; const double eps=1e-;
int n,m,x,c[N<<]; double y,mx[N<<]; int que(int x,int L,int R,double k){
if (k-mx[x]>-eps) return ;
if (L==R) return c[x];
int mid=(L+R)>>;
if (k-mx[ls]>-eps) return que(rs,mid+,R,k);
else return que(ls,L,mid,k)+c[x]-c[ls];
} void mdf(int x,int L,int R,int pos,double k){
if (L==R){ mx[x]=k; if (fabs(k)<eps) c[x]=; else c[x]=; return; }
int mid=(L+R)>>;
if (pos<=mid) mdf(ls,L,mid,pos,k); else mdf(rs,mid+,R,pos,k);
mx[x]=max(mx[ls],mx[rs]); c[x]=c[ls]+que(rs,mid+,R,mx[ls]);
} int main(){
scanf("%d%d",&n,&m);
for (int i=; i<=m; i++) scanf("%d%lf",&x,&y),mdf(,,n,x,y/x),printf("%d\n",c[]);
return ;
}
[Luogu P4198]楼房重建(线段树)的更多相关文章
- luogu P4198 楼房重建——线段树
题目大意: 小A在平面上(0,0)点的位置,第i栋楼房可以用一条连接(i,0)和(i,Hi)的线段表示,其中Hi为第i栋楼房的高度.如果这栋楼房上任何一个高度大于0的点与(0,0)的连线没有与之前的线 ...
- 洛谷P4198 楼房重建(线段树)
题意 题目链接 Sol 别问我为什么发两遍 就是为了骗访问量 这个题的线段树做法,,妙的很 首先一个显然的结论:位置\(i\)能被看到当且仅当\(\frac{H_k}{k} < \frac{H_ ...
- Luogu P4198 楼房重建 (李超线段树)
题目 传送门 题解 首先转化成到(0,0)(0,0)(0,0)的斜率. 那么就是求多少个点是前缀最大值. 做法是线段树,用gao(i,x)gao(i,x)gao(i,x)表示在iii区间内,之前最大值 ...
- Luogu P4198 楼房重建 分块 or 线段树
思路:分块 提交:2次(第一次的求解有问题) 题解: 设块长为$T$,我们开$N/T$个单调栈,维护每一块的上升斜率. 修改时暴力重构整个块,$O(T)$ 求解时记录一个最大斜率$lst$,然后块内二 ...
- luogu题解P4198楼房重建--线段树神操作
题目链接 https://www.luogu.org/problemnew/show/P4198 分析 一句话题意,一条数轴上有若干楼房,坐标为\(xi\)的楼房有高度\(hi\),那么它的斜率为\( ...
- [BZOJ29957] 楼房重建 - 线段树
2957: 楼房重建 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 3294 Solved: 1554[Submit][Status][Discus ...
- bzoj 2957: 楼房重建 线段树
2957: 楼房重建 Time Limit: 10 Sec Memory Limit: 256 MB[Submit][Status][Discuss] Description 小A的楼房外有一大片施 ...
- 【题解】Luogu P4198 楼房重建
原题传送门 根据斜率来建线段树,线段树维护区间最大斜率以及区间内能看见的楼房的数量(不考虑其他地方的原因,两个节点合并时再考虑) 细节见程序 #include <bits/stdc++.h> ...
- bzoj 2957: 楼房重建 ——线段树
Description 小A的楼房外有一大片施工工地,工地上有N栋待建的楼房.每天,这片工地上的房子拆了又建.建了又拆.他经常无聊地看着窗外发呆,数自己能够看到多少栋房子. 为了简化问题,我们考虑这些 ...
随机推荐
- bzoj
准确率爆棚啊,然而
- 你是否彻底了解margin属性?
写css,你少不了与margin打交道.你真的了解margin吗?你知道margin有什么特性吗?你知道什么是垂直外边距合并?margin在块元素.内联元素中的区别?什么时候该用padding而不是m ...
- jw player学习笔记----跨域请求
需求来源:播放器皮肤文件请求不到,被限制了. 参考官网解决方案: http://www.longtailvideo.com/support/jw-player/28844/crossdomain-fi ...
- 栈与递归的实现(Hanoi塔问题等等)
函数中有直接或间接地调用自身函数的语句,这样的函数称为递归函数.递归函数用 得好,可简化编程工作.但函数自己调用自己,有可能造成死循环.为了避免死循环,要 做到两点: (1) 降阶.递归函数虽然调用自 ...
- Nginx各项配置的含义
#user nobody; #配置用户或者组,默认为nobody nobody worker_processes 4; #允许生成的进程数,默认为1 worker_cpu_affinity 00000 ...
- 【poj3420】递推式转矩阵乘法
历史性的时刻!!! 推了一晚上!和hyc一起萌萌哒地推出来了!! 被摧残蹂躏的智商啊!!! 然而炒鸡高兴!! (请不要介意蒟蒻的内心独白..) 设a[i]为扫到第i行时的方案数. 易知,对于一行1*4 ...
- 「6月雅礼集训 2017 Day2」A
[题目大意] 给出一棵树,求有多少对点(u,v)满足其路径上不存在两个点a,b满足(a,b)=1 n<=10^5 [题解] 考虑找出所有不符合的点对,共有n*ln(n)对,他们要么是祖先-> ...
- DotNet 学习笔记 应用状态管理
Application State Options --------------------------------------------------------------------- *Htt ...
- bzoj 1002 找规律(基尔霍夫矩阵)
网上说的是什么基尔霍夫矩阵,没学过这个,打个表找下规律,发现 w[i]=3*w[i-1]-w[i-2]+2; 然后写个高精直接递推就行了 //By BLADEVIL var n :longint; a ...
- 培训补坑(day10:双指针扫描+矩阵快速幂)
这是一个神奇的课题,其实我觉得用一个词来形容这个算法挺合适的:暴力. 是啊,就是循环+暴力.没什么难的... 先来看一道裸题. 那么对于这道题,显然我们的暴力算法就是枚举区间的左右端点,然后通过前缀和 ...