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

/*
poj 2528 Mayor's posters
线段树 + 离散化

离散化的理解:
  给你一系列的正整数, 例如 1, 4 , 100, 1000000000, 如果利用线段树求解的话,很明显
  会导致内存的耗尽。所以我们做一个映射关系,将范围很大的数据映射到范围很小的数据上
  1---->1  4----->2  100----->3  1000000000----->4
  这样就会减少内存一些不必要的消耗
  建立好映射关系了,接着就是利用线段树求解
*/
#include<iostream>
#include<cstdio>
#include<queue>
#include<cstring>
#include<algorithm>
#define N 10000010
#define M 10005
using namespace std;
class EDGE{
public:
   int ld, rd;
};
int tree[M*16];//一共有M*2个端点,一个线段映射到四个点,左右端点, 左端点-1, 右端点+1, 数组的大小是线段树最底层数据个数的4倍
EDGE edge[M];
int p[M*4];
int hash[N];
int n;

int insert(int p, int lr, int rr, int ld, int rd){

    if(tree[p] && lr<=ld && rd<=rr)//如果当前的区间[ld, rd]被包含在[lr, rr]中,而且[lr, rr]的区间已经被覆盖
       return 1;
    else if(lr==ld && rr==rd){
        tree[p]=1;
        return 0;
    }
    else{
        int mid=(lr+rr)>>1;
        int f1, f2, f3, f4;
        if(mid>=rd)
           f1=insert(p<<1, lr, mid, ld, rd);
        else if(mid<ld)
           f2=insert(p<<1|1, mid+1, rr, ld, rd);
        else{
       f3=insert(p<<1, lr, mid, ld, mid);
       f4=insert(p<<1|1, mid+1, rr, mid+1, rd);
        }
    tree[p]=tree[p<<1] && tree[p<<1|1];//两个子树都被覆盖的时候父类才会被覆盖
    if(mid>=rd)
       return f1;
    else if(mid<ld)
       return f2;
    else return f3 && f4;
    }
}
/*
3
1 10
1 3
6 10
如果将一个线段离散化成两个点,输出地结果是2
如果是四个节点,输出的结果就是3
而正确的结果就是3
*/ 

int main(){
   int t, i, nm;
   scanf("%d", &t);
   while(t--){
      int maxR=0;
      scanf("%d", &n);
      for(i=0; i<n; ++i){
            scanf("%d%d", &edge[i].ld, &edge[i].rd);
            p[maxR++]=edge[i].ld-1;
            p[maxR++]=edge[i].ld;
            p[maxR++]=edge[i].rd;
            p[maxR++]=edge[i].rd+1;
      }
      sort(p, p+maxR);
      maxR=unique(p, p+maxR)-p;//元素去重
      for(i=0, nm=0; i<maxR; ++i){
          hash[p[i]]=++nm;
      }
      memset(tree, 0, sizeof(tree));//初始值是所有的点都没有被覆盖
      int ans=0;
      for(i=n-1; i>=0; --i){//由外向里看真是个不错的主意
            if(!insert(1, 1, nm, hash[edge[i].ld], hash[edge[i].rd]))
               ++ans;
      }
      printf("%d\n", ans);
   }
   return 0;
}
时间: 2024-09-15 20:40:12

poj 2528 Mayor's posters(线段树+离散化)的相关文章

poj 2528 Mayor&#039;s posters

点击打开链接poj2528 思路:离散化+线段树成段更新 分析: 1 首先这一题的数据是错误的,这题的区间的最大值为10000000,如果我们按照正常的线段树的思路去做的话肯定是会超内存和超时的. 2 所以我们应该考虑离散化,我们把区间离散成集中的区间.但是这个地方会有个问题 给出下面两个简单的例子应该能体现普通离散化的缺陷: 例子一:1-10 1-4 5-10 例子二:1-10 1-4 6-10 普通离散化后都变成了[1,4][1,2][3,4] 线段2覆盖了[1,2],线段3覆盖了[3,4]

poj 1171 Picture 线段树

     经典的线段树问题,看了好久才看懂      解法很简单,按y坐标从小到大,依次扫描每条线段,每次利用线段树记录当前图形在x轴上的投影,然后用这次投影减去上次就是x轴上变化量,而y轴,因为按y轴枚举,只需要用y的差值乘以2再乘以当前线段数即可.      而线段树的处理就是遇到下边是加入线段树,遇到上边删去及记录当前投影长度,及投影分段情况      看到网上还有种括号匹配的方法,离散化后枚举,复杂度n^2 /* author:jxy lang:C/C++ university:Chin

POJ 3468 线段树 区间更新区间查询

题意:给出一段数列,任意区间加上一个整数v,任意区间查询.给出N个数,Q次操作. 线段树的题,延迟指针不更新到底就行了. #include <iostream> #include<cstdio> #include<cstring> #include<algorithm> using namespace std; #define maxn 100005 long long sum[maxn<<2],col[maxn<<2]; void

【转】POJ 3264 线段树解法

每个算法都是数学家或者计算机学家多年研究的结果,不是我自己臆造的,所以学习一个新算法的最佳方法还是看写的好的代码. 按照惯例,我就粘贴一个网上写的很好的帖子... 原文地址:http://ip96cns.blog.163.com/blog/static/17009519220112525029459/ 线段树的查找:(感谢高手指点,虽然只是一点却让我把线段树的内容理顺了)这里是我的一些总结吧           一个线段树的结点表示一个区间,同时结点中保存所需要的信息.          如果询

线段树(Segment Tree)

1.概述 线段树,也叫区间树,是一个完全二叉树,它在各个节点保存一条线段(即"子数组"),因而常用于解决数列维护问题,基本能保证每个操作的复杂度为O(lgN). 线段树是一种二叉搜索树,与区间树相似,它将一个区间划分成一些单元区间,每个单元区间对应线段树中的一个叶结点. 对于线段树中的每一个非叶子节点[a,b],它的左儿子表示的区间为[a,(a+b)/2],右儿子表示的区间为[(a+b)/2+1,b].因此线段树是平衡二叉树,最后的子节点数目为N,即整个线段区间的长度. 使用线段树可以

线段树

                                                                          线段树   一 什么是线段树     线段树是一棵二叉树,树中的每一个结点表示了一个区间[a,b].每一个叶子节点表示了一个单位区间.对于每一个非叶结点所表示的结点[a,b],其左儿子表示的区间为[a,(a+b)/2],右儿子表示的区间为[(a+b)/2+1,b] 二 线段树的性质 1 线段树是建立在线段的基础上,每个结点都代表了一条线段[a,b].

POJ3468&amp;#160;线段树-模板题

Description You have N integers, A1, A2, ... , AN. You need to deal with two kinds of operations. One type of operation is to add some given number to each number in a given interval. The other is to ask for the sum of numbers in a given interval. In

线段树和RMQ解析和模板

这几天在看RMQ的题目,但是很多RMQ的题目也可以用线段树解决... 看来两者之间有很多关系,那就都好好看吧...下面先贴出一个大牛对两者的解释.. RMQ (Range Minimum/Maximum Query)问题是指: 对于长度为n的数列A,回答若干询问RMQ(A,i,j)(i,j<=n),返回数列A中下标在[i,j]里的最小(大)值,也就是说,RMQ问题是指求区间最值的问题  主要方法及复杂度(处理复杂度和查询复杂度)如下:  1.朴素(即搜索) O(n)-O(n)  2.线段树(se

hdu 4614 Vases and Flowers 线段树

    比赛最后40分钟写的,线段树加二分,思路写法完全没问题,但最后提交了2次都WA了,回来后发现是模板更新lazy把0当做更新过的了,但是应该是1 /* author:jxy lang:C/C++ university:China,Xidian University **If you need to reprint,please indicate the source** */ #include <iostream> #include <cstdio> #include <