题意:给一个数组,每次查询输出区间内不重复数字的和。

这是3xian教主的题。

用前缀和的思想可以轻易求得区间的和,但是对于重复数字这点很难处理。在线很难下手,考虑离线处理。

将所有查询区间从右端点由小到大排序,遍历数组中的每个数字,每次将该数字上次出现位置的值在树状数组中改为0,再记录当前位置,在树状数组中修改为当前的数值。这样可以保证在接下来的查询中该数字只出现了一次。这是贪心的思想,只保留最可能被以后区间查询的位置。如果当前位置是某个查询区间的右端点,这时候就可以查询了。最后再根据查询区间的编号排序输出即可了。

注意树状数组查询0位置会出现死循环。

HDU上long long 需要使用I64d。

 #include <iostream>
 #include <cstdio>
 #include <cstring>
 #include <vector>
 #include <map>
 #include <algorithm>
 #define ll long long
 #define MAXN 100005
 using namespace std;
 int n;
 map<int,int> pos;
 struct Segment
 {
     int num,left,right;
     ll ans;
     Segment(,,):num(a),left(b),right(c)
     {
         ans=;
     }
     bool operator <(const Segment &p) const
     {
         return right<p.right;
     }
 };
 bool cmp(Segment a,Segment b)
 {
     return a.num<b.num;
 }
 struct BIT
 {
     ll dat[MAXN];
     int lowbit(int x)
     {
         return -x&x;
     }
     void clear()
     {
         memset(dat,,sizeof(dat));
     }
     void add(int x,ll val)
     {
         while(x<=n)
         {
             dat[x]+=val;
             x+=lowbit(x);
         }
     }
     ll sum(int x)
     {
         ll s=;
         )
         {
             s+=dat[x];
             x-=lowbit(x);
         }
         return s;
     }
     void modify(int x,ll val)
     {
         ) return ;
         ll t=sum(x)-sum(x-);
         add(x,-t+val);
     }
 };
 int arr[MAXN];
 vector<Segment> vec;
 BIT tree;
 int main()
 {
     int T;
     scanf("%d",&T);
     while(T--)
     {
         scanf("%d",&n);
         pos.clear();
         ; i<=n; ++i)
             scanf("%d",&arr[i]);
         int q;
         scanf("%d",&q);
         vec.clear();
         ; i<=q; ++i)
         {
             int x,y;
             scanf("%d%d",&x,&y);
             vec.push_back(Segment (i,x,y));
         }
         sort(vec.begin(),vec.end());
         tree.clear();
         ,j=; i<=n&&j<vec.size(); ++i)
         {
             tree.modify(pos[arr[i]],);
             tree.modify(i,arr[i]);
             pos[arr[i]]=i;
             while(j<vec.size()&&i==vec[j].right)
             {
                 vec[j].ans=tree.sum(vec[j].right)-tree.sum(vec[j].left-);
                 j++;

             }
         }
         sort(vec.begin(),vec.end(),cmp);
         ; i<vec.size(); ++i)
             printf("%I64d\n",vec[i].ans);
     }
     ;
 }

HDU 3333 - Turing Tree (树状数组+离线处理+哈希+贪心)的更多相关文章

  1. HDU 3333 Turing Tree --树状数组+离线处理

    题意:统计一段序列[L,R]的和,重复元素只算一次. 解法:容易看出在线做很难处理重复的情况,干脆全部讲查询读进来,然后将查询根据右端点排个序,然后离散化数据以后就可以操作了. 每次读入一个数,如果这 ...

  2. HDU 3333 Turing Tree (树状数组)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3333 题意就是询问区间不同数字的和. 比较经典的树状数组应用. //#pragma comment(l ...

  3. HDU3333 Turing Tree 树状数组+离线处理

    Turing Tree Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total ...

  4. HDU 3333 | Codeforces 703D 树状数组、离散化

    HDU 3333:http://acm.hdu.edu.cn/showproblem.php?pid=3333 这两个题是类似的,都是离线处理查询,对每次查询的区间的右端点进行排序.这里我们需要离散化 ...

  5. Necklace HDU - 3874 (线段树/树状数组 + 离线处理)

    Necklace HDU - 3874  Mery has a beautiful necklace. The necklace is made up of N magic balls. Each b ...

  6. HDU 3874 Necklace (树状数组 | 线段树 的离线处理)

    Necklace Time Limit: 15000/5000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total S ...

  7. 2016 Multi-University Training Contest 5 1012 World is Exploding 树状数组+离线化

    http://acm.hdu.edu.cn/showproblem.php?pid=5792 1012 World is Exploding 题意:选四个数,满足a<b and A[a]< ...

  8. SPOJ DQUERY树状数组离线or主席树

    D-query Time Limit: 227MS   Memory Limit: 1572864KB   64bit IO Format: %lld & %llu Submit Status ...

  9. D-query SPOJ 树状数组+离线

    D-query SPOJ 树状数组+离线/莫队算法 题意 有一串正数,求一定区间中有多少个不同的数 解题思路--树状数组 说明一下,树状数组开始全部是零. 首先,我们存下所有需要查询的区间,然后根据右 ...

随机推荐

  1. Oct22 实例测试

    实例: http://www.runoob.com/jsp/jsp-form-processing.html http://www.runoob.com/jsp/jsp-writing-filters ...

  2. JQUERY学习(壹)

    一.jQuery的引言 1.jQuery框架:对JavaScript的封装,简化js开发 2.jQuery框架的好处: 1)语法简单 js中:document.getElementById(" ...

  3. php面向对象Object

    1.创建类 class 类名{ private 私有变量 只能本类的内部使用 protected 受保护的变量 本类和子类的内部 public 公开的变量 都可以使用 一般属性都设为私有 一般函数都是 ...

  4. 20145218 《Java程序设计》第三周学习总结

    20145218 <Java程序设计>第三周学习总结 教材学习内容总结 定义类 编写程序要产生对象就要先定义类.类是对象的设计图,对象是类的实例.类定义时使用class关键词,建立实例时, ...

  5. hive数据库的一些应用

    1.创建表格create table usr_info(mob string,reason string,tag string) row format delimited fields termina ...

  6. 小而美的js程序

    1.获取数字数组最小值的索引 function _getMinKey(arr) { var a = arr[0]; var b = 0; for (var k in arr) { if (arr[k] ...

  7. 第二周 WBS、NABCD查阅

    WBS WBS:工作分解结构(Work Breakdown Structure) 创建WBS:创建WBS是把项目可交付成果和项目工作分解成较小的,更易于管理的组成部分的过程. WBS是项目管理重要的专 ...

  8. android 模拟器 使用键盘的配置

    1 打开 android Manageer , 克隆一个设备 2.

  9. javaSE基础之记事本编程

    首先安装好jdk和jre,之后进行如下操作: 1. 将代码记事本---->cmd--->javac 文件名.java ----->java 文件名 如图: 2. 关于记事本文件属性的 ...

  10. DOI EXCEL显示报表

    我这个是比较不规则的数据填充 1.程序开头,定义一个工作区,存对应单元格的值: BEGIN OF TY_EXCEL, C031() TYPE C, C032() TYPE C, C033() TYPE ...