网上还有用unique函数和lowerbound函数离散的方法,可以百度搜下题解就有。

这里给出介绍unique函数的链接:http://www.cnblogs.com/zhangshu/archive/2011/07/23/2115090.html

#include <iostream>
#include <stdio.h>
#include <string.h>
#include <algorithm>
#include <set>
/*
题意:
给出一条直线上的矩形左右坐标和高度,求所有矩形并的面积。
思路:
这里处理没有跟以前的线段树一样,这里每个叶子节点代表的是区间[a,a+1),而不像以前的代表一个点(a,a);
需要建立基本单位为线段的线段树,建法与以点为单位的线段树就一点点不同,具体见代码。这样可以方便在离散化后求一段区间间的长度 由于A、B的值太大,需要离散。且不可能开那么大的数组,也就不能通过A、B直接获取他们的映射值。
而只能通过他们映射的值来获取对应的A、B,用二分查找。
由于最后求并的面积,为方便起见,可以先将矩形按高度从小到大排序,后面的覆盖前面的即可。 在discuss里别人写的几个注意点:
总算有点收获,总结以下几点
1、离散的时候要用二分来查找映射值
2、数据全部用long long 吧,比较保险
3、数组开大点,因为一共有40000组数据,每组数据两个点,加起来也就是有大约80000个点,所以N要开到80000,存线段树的数组要开到4*N。
4、题目的意思是要算矩形和,也就是说同一个线段上有多个高度的话,取最高的那个,再modify的时候,
可以先对输入的数据根据高度从小到大排序,在接下来的操作中直接覆盖,这样就可以避免一些不必要的麻烦。 */
using namespace std;
const int maxn=;
int n;
int cnt;
long long vk[maxn*]; //vk[i]=A,表示A映射的值为i,要通过二分查找,寻找A对应的映射值
long long val[maxn*];
struct Tree{
long long sum; //该区间的矩阵面积
long long num; //该区间的矩阵高度
bool lazy;
}tree[maxn<<];
struct Node{
long long a,b,h; //区间[a,b],以及高度h
bool operator<(const Node tmp)const{
return h<tmp.h;
}
}node[maxn]; //二分查找k对应的映射值
int binsearch(long long k){
int l=,r=cnt,mid;
while(l<=r){
mid=(l+r)>>;
if(vk[mid]==k)
return mid;
if(vk[mid]<k)
l=mid+;
else
r=mid-;
}
return -; //肯定会查找的到,这里就是摆设
} void build(int rt,int L,int R){
tree[rt].sum=;
tree[rt].num=;
tree[rt].lazy=false;
if(L+==R) //这才是重点啊,建立区间为[a,a+1)的叶子节点
return;
int mid=(L+R)>>;
build(rt<<,L,mid);
build(rt<<|,mid,R); //右儿子:mid~R
} void pushUp(int rt){
tree[rt].sum=(tree[rt<<].sum+tree[rt<<|].sum);
} void pushDown(int rt,int L,int R){
if(tree[rt].lazy){
int lson=rt<<,rson=rt<<|;
int mid=(L+R)>>;
long long l1=vk[L],r1=vk[mid],l2=vk[mid],r2=vk[R],length1,length2;
tree[lson].num=tree[rson].num=tree[rt].num;
length1=r1-l1;length2=r2-l2; //左儿子对应的区间长度和右儿子对应的区间长度
tree[lson].sum=length1*tree[rt].num;
tree[rson].sum=length2*tree[rt].num;
tree[lson].lazy=tree[rson].lazy=true;
tree[rt].lazy=false;
}
}
//L,R是节点rt的区间,l,r是要更新的区间目标
void update(int rt,int L,int R,int l,int r,long long h){
long long left=vk[L],right=vk[R],length;
if(l<=L && R<=r){
tree[rt].num=h;
length=right-left;
tree[rt].sum=h*length;
tree[rt].lazy=true;
return;
}
pushDown(rt,L,R);
int mid=(L+R)>>;
//注意两个if条件里面不能有等号
if(l<mid)
update(rt<<,L,mid,l,r,h);
if(r>mid)
update(rt<<|,mid,R,l,r,h);
//也可以下面注释的语句
/*
if(r<=mid)
update(rt<<1,L,mid,l,r,h);
else if(l>=mid)
update(rt<<1|1,mid,R,l,r,h);
else{
update(rt<<1,L,mid,l,mid,h);
update(rt<<1|1,mid,R,mid,r,h);
}
*/
pushUp(rt);
}
int main()
{
long long a,b,h;
int idx=;
scanf("%d",&n);
for(int i=;i<n;i++){
scanf("%I64d%I64d%I64d",&a,&b,&h);
node[i].a=a;
node[i].b=b;
node[i].h=h;
val[idx++]=a;
val[idx++]=b;
}
//离散化
sort(val,val+idx);
cnt=;
vk[cnt]=val[];
for(int i=;i<idx;i++){
if(val[i]==val[i-])
continue;
vk[++cnt]=val[i]; //只不过这里建立的是映射————key
} build(,,cnt);
sort(node,node+n);
int x,y;
for(int i=;i<n;i++){
//二分查找,获取映射
x=binsearch(node[i].a);
y=binsearch(node[i].b);
//printf("%I64d %I64d: %d %d\n",node[i].a,node[i].b,x,y);
update(,,cnt,x,y,node[i].h);
}
printf("%I64d\n",tree[].sum);
return ;
}

POJ 3277 City Horizon(叶子节点为[a,a+1)的线段树+离散化)的更多相关文章

  1. 离散化+线段树 POJ 3277 City Horizon

    POJ 3277 City Horizon Time Limit: 2000MS Memory Limit: 65536K Total Submissions: 18466 Accepted: 507 ...

  2. [POJ] 3277 .City Horizon(离散+线段树)

    来自这两篇博客的总结 http://blog.csdn.net/SunnyYoona/article/details/43938355 http://m.blog.csdn.net/blog/mr_z ...

  3. poj 3277 City Horizon (线段树 扫描线 矩形面积并)

    题目链接 题意: 给一些矩形,给出长和高,其中长是用区间的形式给出的,有些区间有重叠,最后求所有矩形的面积. 分析: 给的区间的范围很大,所以需要离散化,还需要把y坐标去重,不过我试了一下不去重 也不 ...

  4. POJ 3277 City Horizon(扫描线+线段树)

    题目链接 类似求面积并..2Y.. #include <cstdio> #include <cstring> #include <string> #include ...

  5. POJ 3277 City Horizon

    标题效果: 每间房子的长度给出阴影(在间隔代表)而高度,求阴影总面积. 解题思路:矩形面积并. 以下是代码: #include <set> #include <map> #in ...

  6. POJ 2528 Mayor's posters(线段树+离散化)

    Mayor's posters 转载自:http://blog.csdn.net/winddreams/article/details/38443761 [题目链接]Mayor's posters [ ...

  7. poj 2528 Mayor's posters(线段树+离散化)

    /* poj 2528 Mayor's posters 线段树 + 离散化 离散化的理解: 给你一系列的正整数, 例如 1, 4 , 100, 1000000000, 如果利用线段树求解的话,很明显 ...

  8. poj 2528 Mayor's posters 线段树+离散化技巧

    poj 2528 Mayor's posters 题目链接: http://poj.org/problem?id=2528 思路: 线段树+离散化技巧(这里的离散化需要注意一下啊,题目数据弱看不出来) ...

  9. 【POJ 2482】 Stars in Your Window(线段树+离散化+扫描线)

    [POJ 2482] Stars in Your Window(线段树+离散化+扫描线) Time Limit: 1000MS   Memory Limit: 65536K Total Submiss ...

随机推荐

  1. HDU 1954 Subway tree systems (树的最小表示法)

    题意:用一个字符串表示树,0代表向下走,1代表往回走,求两棵树是否同构. 分析:同构的树经过最小表示会转化成两个相等的串. 方法:递归寻找每一棵子树,将根节点相同的子树的字符串按字典序排列,递归回去即 ...

  2. cadence16.3破解方法

    今天安装cadence16.3,安装了两遍都失败了,百思不得其解,结果总是出现在license上,如下图所示: 后面当然就只能启动demo版了,于是网上找,参照以前古老的法子,终于解决问题! 首先,开 ...

  3. ADO.NET笔记——SQL注入攻击

    相关知识: 可以通过字符串的拼接来构造一个SQL命令字符串,但是SQL命令字符串的拼接确是造成“SQL注入攻击”的重要原因. 考虑下列例子:从ProductCategory表中检索出Name为“Bik ...

  4. javascript 逻辑运算符

    javascript逻辑运算符 NOT(!) AND(&&) OR(||) NOT(!) 返回值的类型一定是Boolean值的 运算数也是Boolean值 返回值是:与相反的boole ...

  5. 转:JAVA中this用法小结

    转:http://blog.sina.com.cn/s/blog_6a6badc90100t8hm.html#SinaEditor_Temp_FontName Java关键字this只能用于方法方法体 ...

  6. 一款jquery编写图文下拉二级导航菜单特效

    一款jquery编写图文下拉二级导航菜单特效,效果非常简洁大气,很不错的一款jquery导航菜单特效. 这款jquery特效适用于很多的个人和门户网站. 适用浏览器:IE8.360.FireFox.C ...

  7. Laravel 5 基础(三)- 向视图传送数据

    我们在Routes.php中新建一个路由 Route::get('about', 'PagesController@about'); 在浏览器中浏览会获得一个错误,错误信息仅仅是一个提示信息,缺少细节 ...

  8. Laravel 5 基础(一)- Laravel入门和新建项目

    此系列文章是 laracasts.com 中的入门系列视频的笔记,我做了一些修改,可以参考此系列文章来学习 Laravel 5.原视频作者是 Jeffrey Way, 在此感谢.本人使用的系统是Mac ...

  9. 从零开始学ios开发(十五):Navigation Controllers and Table Views(中)

    这篇内容我们继续上一篇的例子接着做下去,为其再添加3个table view的例子,有了之前的基础,学习下面的例子会变得很简单,很多东西都是举一反三,稍稍有些不同的内容,好了,闲话少说,开始这次的学习. ...

  10. virtualbox usb连接问题解决

    生命在于折腾... 神奇的liinux... ubuntu 14.04 LTS sudo apt-get install virtualbox -y 然后建好虚拟机之后(windows也好,linux ...