D. Necklace
Time Limit: 5000ms
Memory Limit: 32768KB
64-bit integer IO format: %I64d Java class name: MainMery has a beautiful necklace. The necklace is made up of N magic balls. Each ball has a beautiful value. The balls with the same beautiful value look the same, so if two or more balls have the same beautiful value, we just count it once. We define the beautiful value of some interval [x,y] as F(x,y). F(x,y) is calculated as the sum of the beautiful value from the xth ball to the yth ball and the same value is ONLY COUNTED ONCE. For example, if the necklace is 1 1 1 2 3 1, we have F(1,3)=1, F(2,4)=3, F(2,6)=6.
Now Mery thinks the necklace is too long. She plans to take some continuous part of the necklace to build a new one. She wants to know each of the beautiful value of M continuous parts of the necklace. She will give you M intervals [L,R] (1<=L<=R<=N) and you must tell her F(L,R) of them.
Now Mery thinks the necklace is too long. She plans to take some continuous part of the necklace to build a new one. She wants to know each of the beautiful value of M continuous parts of the necklace. She will give you M intervals [L,R] (1<=L<=R<=N) and you must tell her F(L,R) of them.
Input
The first line is T(T<=10), representing the number of test cases.
For each case, the first line is a number N,1 <=N <=50000, indicating the number of the magic balls. The second line contains N non-negative integer numbers not greater 1000000, representing the beautiful value of the N balls. The third line has a number M, 1 <=M <=200000, meaning the nunber of the queries. Each of the next M lines contains L and R, the query.
For each case, the first line is a number N,1 <=N <=50000, indicating the number of the magic balls. The second line contains N non-negative integer numbers not greater 1000000, representing the beautiful value of the N balls. The third line has a number M, 1 <=M <=200000, meaning the nunber of the queries. Each of the next M lines contains L and R, the query.
Output
For each query, output a line contains an integer number, representing the result of the query.
Sample Input
2 6 1 2 3 4 3 5 3 1 2 3 5 2 6 6 1 1 1 2 3 5 3 1 1 2 4 3 5
Sample Output
3 7 14 1 3 6
解题:离线树状数组,为什么要把y坐标从小到大进行排序呢?因为树状数组的特性所致,右边的节点可能包含左边的节点的值,所以从左往右,不断去掉重复的数值更简便。
离线处理即先一次性把所有询问存储起来。最后一次性回复。用一个数组pre[i]记录元素d[i]距离i最近的一次出现的下标。pre[i] == -1即表示该元素d[i]目前是唯一的,不存在重复元素的。
1 #include <iostream> 2 #include <cstdio> 3 #include <cstring> 4 #include <cmath> 5 #include <algorithm> 6 #include <climits> 7 #include <vector> 8 #include <queue> 9 #include <cstdlib> 10 #include <string> 11 #include <set> 12 #define LL long long 13 #define INF 0x3f3f3f 14 using namespace std; 15 const int maxn = 210000; 16 struct query { 17 int x,y,id; 18 } q[maxn]; 19 LL tree[51000]; 20 int pre[51000],loc[1100000],n,m,d[51000]; 21 LL ans[maxn]; 22 bool cmp(const query &a,const query &b) { 23 return a.y < b.y; 24 } 25 int lowbit(int x) { 26 return x&(-x); 27 } 28 void update(int x,int val) { 29 for(; x <= n; x += lowbit(x)) { 30 tree[x] += val; 31 } 32 } 33 LL sum(int x) { 34 LL ans = 0; 35 for(; x; x -= lowbit(x)) 36 ans += tree[x]; 37 return ans; 38 } 39 int main() { 40 int t,i,j; 41 scanf("%d",&t); 42 while(t--) { 43 memset(loc,-1,sizeof(loc)); 44 memset(tree,0,sizeof(tree)); 45 scanf("%d",&n); 46 for(i = 1; i <= n; i++) { 47 scanf("%d",d+i); 48 pre[i] = loc[d[i]]; 49 loc[d[i]] = i; 50 update(i,d[i]); 51 } 52 scanf("%d",&m); 53 for(i = 1; i <= m; i++) { 54 scanf("%d %d",&q[i].x,&q[i].y); 55 q[i].id = i; 56 } 57 sort(q+1,q+m+1,cmp); 58 int y = 0; 59 for(i = 1; i <= m; i++) { 60 for(j = y+1; j <= q[i].y; j++) { 61 if(pre[j] != -1) update(pre[j],-d[j]); 62 } 63 y = q[i].y; 64 ans[q[i].id] = sum(q[i].y) - sum(q[i].x-1); 65 } 66 for(i = 1; i <= m; i++) { 67 printf("%I64d ",ans[i]); 68 } 69 } 70 return 0; 71 }