Description

  小A的楼房外有一大片施工工地,工地上有N栋待建的楼房。每天,这片工地上的房子拆了又建、建了又拆。他经常无聊地看着窗外发呆,数自己能够看到多少栋房子。
  为了简化问题,我们考虑这些事件发生在一个二维平面上。小A在平面上(0,0)点的位置,第i栋楼房可以用一条连接(i,0)和(i,Hi)的线段表示,其中Hi为第i栋楼房的高度。如果这栋楼房上任何一个高度大于0的点与(0,0)的连线没有与之前的线段相交,那么这栋楼房就被认为是可见的。
  施工队的建造总共进行了M天。初始时,所有楼房都还没有开始建造,它们的高度均为0。在第i天,建筑队将会将横坐标为Xi的房屋的高度变为Yi(高度可以比原来大---修建,也可以比原来小---拆除,甚至可以保持不变---建筑队这天什么事也没做)。请你帮小A数数每天在建筑队完工之后,他能看到多少栋楼房?

Input

  第一行两个正整数N,M
  接下来M行,每行两个正整数Xi,Yi

Output

  M行,第i行一个整数表示第i天过后小A能看到的楼房有多少栋

Sample Input

3 4
2 4
3 6
1 1000000000
1 1

Sample Output

1
1
1
2
数据约定
  对于所有的数据1<=Xi<=N,1<=Yi<=10^9,N,M<=100000
 
 
 
=========华丽丽的分割线============
虽然是一个清华集训的题目,不过还是可做的嘛。。。
一开始的时候自己写了一个程序,然后怎么都没有调出来。
考虑本题,给出一个数列,然后要求支持单点修改以及询问比自己左边所有数都大的数就几个。
考虑线段树,维护出一段中的高度最大值以及别的数都不考虑的情况下(这个一定不能漏)比这样的数有几个。
考虑合并两个线段,高度的最大值是很容易合并的,直接取一个max就可以了。
对于贡献度,我们发现一个线段的左半部分所有满足的数在原来的线段中一定满足,于是我们只需要考虑右半部分。
我们写一个函数calc(node,k)表示node这个线段在左侧有一个大小为k的数的时候内部满足条件的数的个数。
于是发现如果这个线段的左半部分的最大值小于等于k,那么左半部分贡献就是0,直接返回calc(node*2+1,k),
如果左半部分最大值大于k,那么右半部分原本的个数是不会变的,然后再加上calc(node*2,k)就可以了,
时间复杂度O(nlog^2n),
听说这题卡精度,在吕爷爷的帮助下我学会了fraction,代码如下:

 #include <bits/stdc++.h>
#define Maxn 100007
using namespace std;
int read()
{
int x=,f=;char ch=getchar();
while (ch<''||ch>''){if (ch=='-') f=-;ch=getchar();}
while (ch>=''&&ch<=''){x=x*+ch-'';ch=getchar();}
return x*f;
}
struct fraction
{
int dx,dy;
};
bool operator >(fraction a, fraction b)
{
return (1LL*a.dx*b.dy>1LL*a.dy*b.dx);
}
bool operator >=(fraction a, fraction b)
{
return (1LL*a.dx*b.dy>=1LL*a.dy*b.dx);
}
bool operator <(fraction a, fraction b)
{
return (1LL*a.dx*b.dy<1LL*a.dy*b.dx);
}
bool operator <=(fraction a, fraction b)
{
return (1LL*a.dx*b.dy<=1LL*a.dy*b.dx);
}
int n,m;
struct seg
{
int lx,rx,cnt;
fraction hmax;
};
seg tree[Maxn*];
void build(int node, int l, int r)
{
tree[node].lx=l,tree[node].rx=r,tree[node].cnt=;
tree[node].hmax=(fraction){,};
if (tree[node].lx==tree[node].rx) return;
int mid=(l+r)/;
build(node*,l,mid),build(node*+,mid+,r);
}
int calc(int node, fraction h)
{
if (tree[node].hmax<=h) return ;
if (tree[node].lx==tree[node].rx) return ;
if (tree[node*].hmax<=h) return calc(node*+,h);
else return tree[node].cnt-tree[node*].cnt+calc(node*,h);
}
void update(int node, int pos, fraction h)
{
if (tree[node].rx<pos) return;
if (tree[node].lx>pos) return;
if (tree[node].lx==tree[node].rx)
{
tree[node].hmax=h;
tree[node].cnt=;
return;
}
update(node*,pos,h),update(node*+,pos,h);
tree[node].hmax=max(tree[node*].hmax,tree[node*+].hmax);
tree[node].cnt=tree[node*].cnt+calc(node*+,tree[node*].hmax);
}
int main()
{
n=read(),m=read();
build(,,n);
while (m--)
{
int x=read(),y=read();
update(,x,(fraction){y,x});
printf("%d\n",tree[].cnt);
}
return ;
}
 

【数据结构】bzoj2957楼房重建的更多相关文章

  1. BZOJ2957: 楼房重建(线段树&LIS)

    2957: 楼房重建 Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 3727  Solved: 1793[Submit][Status][Discus ...

  2. Bzoj2957 楼房重建

    Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 1516  Solved: 723[Submit][Status][Discuss] Descripti ...

  3. [bzoj2957][楼房重建] (线段树)

    Description 小A的楼房外有一大片施工工地,工地上有N栋待建的楼房.每天,这片工地上的房子拆了又建.建了又拆.他经常无聊地看着窗外发呆,数自己能够看到多少栋房子. 为了简化问题,我们考虑这些 ...

  4. 【分块】bzoj2957 楼房重建

    http://www.cnblogs.com/wmrv587/p/3843681.html ORZ 分块大爷.思路很神奇也很清晰. 把 块内最值 和 块内有序 两种良好的性质结合起来,非常棒地解决了这 ...

  5. 【经典问题】bzoj2957: 楼房重建

    经典问题:动态维护上升子序列长度 进阶问题:[经典问题]#176. 栈 Description 小A的楼房外有一大片施工工地,工地上有N栋待建的楼房.每天,这片工地上的房子拆了又建.建了又拆.他经常无 ...

  6. BZOJ2957 楼房重建 【线段树】

    题目 小A的楼房外有一大片施工工地,工地上有N栋待建的楼房.每天,这片工地上的房子拆了又建.建了又拆.他经常无聊地看着窗外发呆,数自己能够看到多少栋房子. 为了简化问题,我们考虑这些事件发生在一个二维 ...

  7. bzoj2957 楼房重建——线段树

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2957 线段树维护两个值:cnt 能看到的最多楼房数: mx 最大斜率数: 对于一段区间,从左 ...

  8. BZOJ2957: 楼房重建(分块)

    题意 题目链接 Sol 自己YY出了一个\(n \sqrt{n} \log n\)的辣鸡做法没想到还能过.. 可以直接对序列分块,我们记第\(i\)个位置的值为\(a[i] = \frac{H_i}{ ...

  9. bzoj2957楼房重建

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2957 线段树.每个点记录斜率,要一个单增的序列长度(从1开始). 线段树每个点记录自己区间的 ...

随机推荐

  1. OpenCV代码提取:flip函数的实现

    OpenCV中实现图像翻转的函数flip,公式为: 目前fbc_cv库中也实现了flip函数,支持多通道,uchar和float两种数据类型,经测试,与OpenCV3.1结果完全一致. 实现代码fli ...

  2. express操作数据库

    Express 首页 入门 使用指南 API 中文手册 进阶话题 有用的资源 集成数据库 为 Express 应用添加连接数据库的能力,只需要加载相应数据库的 Node.js 驱动即可.这里将会简要介 ...

  3. c++ list_iterator demo

    #include <iostream> #include <list> using namespace std; typedef list<int> Integer ...

  4. OpenCV入门:(二:加载,显示,修改以及保存图片)

    目标: 1.从图片文件打开图片(imread) 2.显示图片(namedWindow和imshow) 3.转换当前图片为灰色图片(cvtColor) 4.保存图片(imwrite) 代码: #incl ...

  5. LeetCode 36——有效的数独

    1. 题目 2. 解答 将数独中数字的 ASCII 码值转化到 0-8 之间作为散列值,建立一个散列表,然后分别逐行.逐列.逐宫(3*3小块)统计每个数字的出现次数,若出现次数大于 1,则数独无效. ...

  6. [比赛总结]ACM div3 G 比赛总结

    这次题目总体感觉和做阅读理解差不多,英文题目读起来相当费劲. 另外,这次比赛整个队伍中我们三个都突出存在的问题就是,把简单问题复杂化,抓不到事物的本质,因此很容易的就被题目误导. 比如C题,明明想到了 ...

  7. [剑指Offer] 30.连续子数组的最大和

    题目描述 HZ偶尔会拿些专业问题来忽悠那些非计算机专业的同学.今天测试组开完会后,他又发话了:在古老的一维模式识别中,常常需要计算连续子向量的最大和,当向量全为正数的时候,问题很好解决.但是,如果向量 ...

  8. Oracle解决索引碎片功能

    我们开始时向一个空的带索引的表中插入大量数据后,是不会产生碎片问题的,但是,数据库经过很长一段时间的增删改查后,难免会出现碎片问题,影响数据库的性能,Oracle对于这一问题有自己的解决方案. 下面介 ...

  9. Codeforces Round #268 (Div. 1) 468D Tree(杜教题+树的重心+线段树+set)

    题目大意 给出一棵树,边上有权值,要求给出一个1到n的排列p,使得sigma d(i, pi)最大,且p的字典序尽量小. d(u, v)为树上两点u和v的距离 题解:一开始没看出来p需要每个数都不同, ...

  10. JAVA 异常处理的认知学习过程

    没有异常处理 学生时代,我编写的java代码中,很少会有try catch.最主要的原因如下: 应用的规模很小 没有不确定因素 代码可控性高 如果规模小,往往就没有复杂的逻辑链路,整个软件的分层也很浅 ...