HDU3874 线段树 + 离线处理

  题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3874 , 线段树(或树状数组) + 离线处理

  下午做了第一道离线处理的题目(HDU4417),多少有点感觉,顺便就把这道题也给做了。



  这道题就是要求某个区间内不重复数的和,自己在网上百度后参考别人的方法才AC的。参考来源:http://www.cnblogs.com/gj-Acit/p/3249827.html

  这道题还是需要先离线处理数据,具体方法:

  ① 先用线段树把树给建立起来;

  ② 先把所有要查询的区间存起来,按照区间的右边界的大小从小到大排序,注意保存下标;

    ③ 对于排序后的每次询问要这样处理:把上次询问区间的右边界到当前询问的右边界中的这些数字进行处理,如果这些数字在之前出现过,则把之前出现的这个数字给删掉(置0即可),然后单点更新线段树;然后查询的话用线段树进行区间求和即可。

  我参考的方法中是这样处理的,即用一个数组vis[]存储数字a[i]第一次出现的位置,后来扫描的时候如果发现 vis[a[i]] != i ,即说明该数字之前出现过。

  最后注意这里用int储存结果会爆掉的,所以要用__int64.

#include <iostream>
#include <cstdio>
#include <vector>
#include <cmath>
#include <string>
#include <string.h>
#include <algorithm>
using namespace std;
#define LL __int64
#define eps 1e-8
#define lson l , m , rt << 1
#define rson m + 1 , r , rt << 1 | 1
const int MOD = 10000007;
const int maxn = 50000 + 5;
const int N = 1000000 + 5;
LL sum[maxn << 2] , ans[maxn << 2];        //sum[]存线段树,ans[]存结果
int vis[N] , a[maxn];    //vis[]第一次出现的位置,a[]存储叶子节点信息
struct Section {        //区间
    int L , R;
    int index;
    bool operator < (const Section tmp) const {
        return R < tmp.R;
    }
} s[maxn << 2];
void PushUp(int rt)
{
    sum[rt] = sum[rt << 1] + sum[rt << 1 | 1];
}
void build()
{
    memset(sum , 0 , sizeof(sum));
    memset(vis , 0 , sizeof(vis));
}
void update(int pos , int x , int l , int r , int rt)
{
    if(l == r) {
        sum[rt] = x;
        return;
    }
    int m = (l + r) >> 1;
    if(pos > m)
        update(pos , x , rson);
    else
        update(pos , x , lson);
    PushUp(rt);
}
LL query(int L , int R , int l , int r , int rt)
{
    if(L <= l && R >= r) {
        return sum[rt];
    }
    int m = (l + r) >> 1;
    if(m < L)
        return query(L , R , rson);
    else if(m >= R)
        return query(L , R , lson);
    else
        return query(L , R , lson) + query(L , R , rson);
}
int main()
{
    int T , i , j , n , m;
    cin >> T;
    while(T--)
    {
        build();
        scanf("%d" , &n);
        for(i = 1 ; i <= n ; i++) {
            scanf("%d" , &a[i]);
            if(!vis[a[i]])        //a[i]第一次出现的位置
                vis[a[i]] = i;
            update(i , a[i] , 1 , n , 1);    //实际是在这里建的树
        }
        scanf("%d" , &m);
        for(i = 1 ; i <= m ; i++) {
            scanf("%d %d" , &s[i].L , &s[i].R);
            s[i].index = i;
        }
        sort(s + 1 , s + m + 1);
        for(i = 1 , s[0].R = 1; i <= m ; i++) {        //此为步骤③
            for(j = s[i - 1].R ; j <= s[i].R ; j++) {
                if(vis[a[j]] != j) {
                    int tmp = vis[a[j]];
                    update(tmp , 0 , 1 , n , 1);    //删掉之前出现的该数字
                    vis[a[j]] = j;                    //删掉后再换位置
                }
            }
            ans[s[i].index] = query(s[i].L , s[i].R , 1 , n , 1);
        }
        for(i = 1 ; i <= m ; i++)
            printf("%I64d\n" , ans[i]);
    }
    return 0;
}
时间: 2024-12-28 11:27:11

HDU3874 线段树 + 离线处理的相关文章

Super Mario(线段树离线区间k值)

以前见过这题,没做出来,知道是离线处理,这次仔细想了下, 首先把出现的高度都map离散化一下,以离散化出来的数目g建树,把每个位置都开俩个vector,一个存以这个位置为L的询问,一个存以这个位置为R的询问. 然后从1-g 进行更新,假如当前i是以第j个区间的开始位置,那么这时就可以询问一下<=p[j].h的个数s,显然这时第J个区间多加的,需要减掉,p[j].sum-=s; 然后更新第i个数,update(a[i],1,g,1);再找到某第k区间是以i结尾的,那么依旧询问一下,得出s,p[k]

线段树+离线 hdu5654 xiaoxin and his watermelon candy

传送门:点击打开链接 题意:一个三元组假设满足j=i+1,k=j+1,ai<=aj<=ak,那么就好的.如今告诉你序列.然后Q次询问.每次询问一个区间[l,r],问区间里有多少个三元组满足要求 思路:刚開始看错题目了,原来三元组是连续3个,这作为bc最后一题也太水了把. . . 先一遍预处理.把连续3个满足条件的找出来,放到还有一个数组里排序去重,用这个数组来给三元组哈希.再扫一遍给三元组在之前那个排序好的数组里二分一下得到下标,大概就是哈希一下,用一个数字来表示. 之后的查询.事实上就是.在

gcd(线段树离线处理)——HDU 4630

对应HDU题目:点击打开链接 No Pain No Game Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Submission(s): 1801    Accepted Submission(s): 770 Problem Description Life is a game,and you lose it,so you suicide. But you can

SPOJ--K-query (线段树离线) 离线操作解决一下问题

K-query Given a sequence of n numbers a1, a2, ..., an and a number of k- queries. A k-query is a triple (i, j, k) (1 ≤ i ≤ j ≤ n). For each k-query (i, j, k), you have to return the number of elements greater than k in the subsequence ai, ai+1, ...,

51nod 1463 找朋友(线段树+离线处理)

http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1463 题意: 思路: 好题! 先对所有查询进行离线处理,按照右区间排序,因为k一共最多只有10个,所有在该区间内的B数组,每次枚举K值,通过这样的方式来得到另外一个B值.但是这样得到的B值它在B数组中的位置必须在当前数的左边.如下图:(j为当前数在B数组中的位置,pos为计算得到的另一个B值在数组中的位置) 这两个数的和记录在pos中,这里pos的位置必须在j的左边,假

玲珑oj 1117 线段树+离线+离散化,laz大法

1117 - RE:从零开始的异世界生活 Time Limit:1s Memory Limit:256MByte Submissions:438Solved:68 DESCRIPTION 486到了异世界,看到了一群可爱的妹子比如蕾姆啊,艾米莉亚啊,拉姆啊,白鲸啊,怠惰啊等等!有一天膜女告诉486说她的能力可能不能再用了,因为膜女在思考一个数据结构题,没心情管486了.486说我来帮你做,膜女说你很棒棒哦! 给一个集合,最开始为空(不是数学上的集合)五个操作: 1.插入x2.把小于x的数变成x3

HDU4417 线段树 + 离线处理

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4417 , 线段树(或树状数组) + 离线处理 最近看了几道线段树的题都是需要离线处理数据的,正好这块比较手生,就练练了.  这道题主要的地方就是离线处理数据,具体想法: ① 先把所有位置的高度都存下来,然后排序,注意保存下标: ② 把所有询问存下来,然后按照询问的高度进行排序,同注意保存下标: ③ 对于排序后的每次询问的处理:由于每个位置的高度都已经存了下来并且进行了排序,所以可以按照顺序将每个点插

HDU 4638-Group(线段树+离线处理)

题意: 给n个编号,m个查询每个查询l,r,求下标区间[l,r]中能分成标号连续的组数(一组内的标号是连续的) 分析: 我们认为初始,每个标号为一个组(线段树维护区间组数),从左向右扫序列,当前标号,要考虑和他相邻的标号的位置,若前面位置出现了和它相邻的标号, 则前面位置组数减一(因为可以合并成一组),查询区间离线处理,保证了查询的正确. #include <map> #include <set> #include <list> #include <cmath&g

HDU 4630-No Pain No Game(线段树+离线处理)

题意: 给你n个数的序列a,q个询问,每个询问给l,r,求在下标i在[l,r]的区间任意两个数的最大公约数中的最大值 分析: 有了hdu3333经验,我们从左向右扫序列,如果当前数的约数在前面出现过,那这个约数可能就是最大的答案.所以我们枚举当前数的所有约数,用线段树维护区间最大值,查询序列离线处理保证查询的正确. #include <map> #include <set> #include <list> #include <cmath> #include