zoukankan      html  css  js  c++  java
  • 【hdu】Mayor's posters(线段树区间问题)

    须要离散化处理,线段树的区间改动问题。

    须要注意的就是离散化的时候,由于给的数字是一段单位长度,所以须要特殊处理(由于线段的覆盖和点的覆盖是不一样的)

    比方:(1,10)(1,4) (6,10)

    离散化之后是 1 , 4 , 6 , 10 分别离散为 1 2 3 4

    覆盖的时候先覆盖1 4 之后覆盖了1 2 之后覆盖了 2 3,结果为2

    可是实际上应该是3

    13450359 201301052100 2528 Accepted 1140K 985MS C++ 1960B 2014-09-17 15:42:33

    985MS险过。

    #include<cstdio>
    #include<cstring>
    #include<iostream>
    #include<algorithm>
    #include<vector>
    #include<queue>
    #include<map>
    using namespace std;
    typedef long long LL;
    const int maxn = 11111;
    int n,m,ans;
    bool vis[maxn];
    int arr[maxn << 2];
    int arr_l[maxn];
    int arr_r[maxn];
    int tr[maxn << 4];
    void UpDate(int l,int r,int value,int L,int R,int pos){
        if(l <= L && R <= r){
            tr[pos] = value;
            return ;
        }
        if(tr[pos] != -1){
            tr[pos << 1]     = tr[pos];
            tr[(pos << 1)|1] = tr[pos];
            tr[pos] = -1;
        }
        int m = (L + R) >> 1;
        if(l <= m) UpDate(l,r,value,L,m,pos << 1);
        if(r  > m) UpDate(l,r,value,m + 1,R,(pos << 1)|1);
        return ;
    }
    void Query(int L,int R,int pos){
        if(tr[pos] != -1){
            int e = tr[pos];
            if(!vis[e]){
                vis[e] = true;
                ans ++;
            }
            return ;
        }
        if(L == R) return ;
        int m = (L + R) >> 1;
        Query(L,m,pos << 1);
        Query(m + 1,R,(pos << 1)|1);
        return ;
    }
    int main(){
        int T;
        scanf("%d",&T);
        while(T--){
            memset(vis,false,sizeof(vis));
            memset(tr,-1,sizeof(tr));
            scanf("%d",&m);
            int size = 0;
            for(int i = 0 ; i < m ; i++){
                scanf("%d%d",&arr_l[i],&arr_r[i]);
                arr[size ++] = arr_l[i];
                arr[size ++] = arr_r[i];
            }
            sort(arr,arr + size);
            n = unique(arr,arr + size) - arr;   //去反复
            for(int i = n - 1 ; i > 0 ; i--)   //区间转化成点
                if(arr[i] != arr[i - 1] + 1)
                    arr[n ++] = arr[i - 1] + 1;
            sort(arr,arr + n);
            ans = 0;
            for(int i = 0 ; i < m ; i++){
                int l = find(arr,arr + n,arr_l[i]) - arr;  //二分查找
                int r = find(arr,arr + n,arr_r[i]) - arr;  //二分查找
                UpDate(l,r,i,0,n - 1,1);
            }
            Query(0,n - 1,1);
            printf("%d
    ",ans);
        }
        return 0;
    }
    

  • 相关阅读:
    TCP—为什么是AIMD?
    虚拟机是怎么实现的?
    漫谈linux文件IO
    关于大型网站技术演进的思考
    大公司里怎样开发和部署前端代码
    spawn-fcgi 代码介绍
    使用python传参form-data格式的txt请求接口
    实战scrapy抓取站长图片数据
    通过requests和lxml模块对网站数据进行爬取
    centos7.5下安装jenkins
  • 原文地址:https://www.cnblogs.com/jhcelue/p/7199803.html
Copyright © 2011-2022 走看看