(树状数组+离线查询)HDU 4417 - Super Mario

题意:

给定一个数列,最多10万次查询l到r不超过h的数字的个数。

分析:

唉,太菜啦。

在线做法应该比较明显,区间维护平衡树,用线段树套平衡树,或者分块套平衡树,应该都能A,但是没试过,只是BB,如有错误欢迎指正。

其实最方便的做法离线做法,太巧妙啦。

把数列按升序排列,把所有查询按h升序排列。

每次查询把比h的小的位置标记为1,查询用bit的sum(r)-sum(l-1)即可

因为都是单调的,所以很方便。

其实很多没有修改的区间问题都可以转化成离线问题。

甚至于一些带修改的题都可以离线。

这些题关键在于如何转化,也关键在于做题者的脑洞。

代码:

 1 #include <cstdio>
 2 #include <cstring>
 3 #include <algorithm>
 4 #include <iostream>
 5 #include <vector>
 6
 7
 8 using namespace std;
 9
10 const int inf = 0x3f3f3f3f;
11 const int maxn = 100010;
12
13 int bit[maxn];
14 int t, n, q;
15
16 int sum(int i) {
17     int s = 0;
18     while(i > 0) {
19         s += bit[i];
20         i -= i & -i;
21     }
22     return s;
23 }
24
25 void add(int i, int x) {
26     while(i <= n) {
27         bit[i] += x;
28         i += i & -i;
29     }
30 }
31
32 struct Q {
33     int l, r, h;
34     int index;
35 } query[maxn];
36
37 bool cmp(Q a, Q b) {
38     return a.h < b.h;
39 }
40
41 pair<int, int> a[maxn];
42 int ans[maxn];
43
44 int main() {
45     scanf("%d", &t);
46     int kase = 0;
47     while(t--) {
48         memset(bit, 0, sizeof(bit));
49         scanf("%d%d", &n, &q);
50         for(int i = 1; i <= n; i++) {
51             scanf("%d", &a[i].first);
52             a[i].second = i;
53         }
54         for(int i = 0; i < q; i++) {
55             scanf("%d%d%d", &query[i].l, &query[i].r, &query[i].h);
56             query[i].l++, query[i].r++;
57             query[i].index = i;
58         }
59         sort(query, query + q, cmp);
60         sort(a + 1, a + n + 1);
61         int left = 1;
62         for(int i = 0; i < q; i++) {
63             while(left <= n && a[left].first <= query[i].h) {
64                 add(a[left].second, 1);
65                 left++;
66             }
67             int r = sum(query[i].r);
68             int l = sum(query[i].l - 1);
69             ans[query[i].index] = r - l;
70         }
71         printf("Case %d:\n", ++kase);
72         for(int i = 0; i < q; i++) {
73             printf("%d\n", ans[i]);
74         }
75     }
76
77
78
79     return 0;
80
81 }
时间: 2024-10-10 07:54:46

(树状数组+离线查询)HDU 4417 - Super Mario的相关文章

HDU 3333 Turing Tree 树状数组 离线查询

题意: 给你一个数列,然后有n个查询,问你给定区间中不同数字的和是多少. 思路还是比较难想的,起码对于蒟蒻我来说. 将区间按照先右端点,后左端点从小到大排序之后,对于每个查询,我只要维护每个数字出现的最后一次就可以了(这个结论稍微想一下就可以证明是正确的). 然后就是简单的点更新,区间求和问题了- #include <cstdio> #include <cstring> #include <iostream> #include <map> #include

【树状数组+离线查询】HDU 3333 Turing Tree

https://www.bnuoj.com/v3/contest_show.php?cid=9149#problem/H [题意] 给定一个数组,查询任意区间内不同数字之和. (n<=30000,Q<=100000,每个数字<=1 000 000 000) [思路] 要算任意区间内不同数字之和,如果我们从左到右依次处理,每次只保留最右边出现的数字,处理以当前数字为右端点的查询,就能做到"不同数字之和",即不重不漏.因此我们要离线处理查询,按记录每个数作为右端点的所有查

HDU 4417 Super Mario ( 超级马里奥 + 主席树 + 线段树/树状数组离线处理 + 划分树 )

HDU 4417 - Super Mario ( 主席树 + 线段树/树状数组离线处理 + 划分树 ) 这道题有很多种做法,我先学习的是主席树.后面陆续补上线段树离线和划分树 题目大意就是给定一个区间给定一个数列,每次要求你查询区间[L,R]内不超过K的数的数量 主席树做法: 最基本的是静态第k大,这里是求静态的 <= K,差不多,在Query操作里面需要修改修改 先建立size棵主席树,然后询问的时候统计的是 第R棵主席树中[1,K]的数量 - 第L-1棵主席树中[1,K]的数量 注意这里下标

hdu 3333 树状数组+离线处理

http://acm.hdu.edu.cn/showproblem.php?pid=3333 不错的题,想了很久不知道怎么处理,而且答案没看懂,然后找个例子模拟下别人的代码马上懂了---以后看不懂的话就拿个例子模拟下别人的代码 举个例子:1 3 3 5 3 5 查询 a, 2 4 b, 2 5 最初是这么想的:对于a查询,倘若把第二个数第三个数变成1个3,那么到b查询,又出现了两个3,再做处理似乎还是O(n),而且如果先出现2,5查询,后出现2,4查询,那么还需要把删除的数补回来.....o(╯

hdu 4368 树状数组 离线维护

http://acm.hdu.edu.cn/showproblem.php?pid=4638 Problem Description There are n men ,every man has an ID(1..n).their ID is unique. Whose ID is i and i-1 are friends, Whose ID is i and i+1 are friends. These n men stand in line. Now we select an interv

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

Necklace HDU - 3874 Mery 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

HDU 3333 Turing Tree(树状数组离线处理)

HDU 3333 Turing Tree 题目链接 题意:给定一个数组,每次询问一个区间,求出这个区间不同数字的和 思路:树状数组离线处理,把询问按右端点判序,然后用一个map记录下每个数字最右出现的位置,因为一个数字在最右边出现,左边那些数字等于没用了,利用树状数组进行单点修改区间查询即可 代码: #include <cstdio> #include <cstring> #include <algorithm> #include <map> using n

13杭州区域赛现场赛Rabbit Kingdom(树状数组+离线)

题意:给你一个长度数列,再给你m个询问(一个区间),问你在这个区间里面有多少个数与其他的数都互质. 解题思路:你看这种类型的题目都可以肯定这是 离线+树状数组(线段树).主要就是他的更新信息.这里我的处理是先把1-200000(每个数的范围)数里面所有的质因子求出来.然后从后往前遍历数组.会出现以下几种情况 1.a[k]的质因子在后面出现过而[更新标记] 和[被更新标记] 都为假 2.a[k]的质因子在后面出现过的那个位置 I   [更新标记]为 真 . 3.a[k]的质因子在后面出现过且那个位

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

1012 World is Exploding 题意:选四个数,满足a<b and A[a]<A[b]   c<d and A[c]>A[d] 问有几个这样的集合 思路: 树状数组+离线化 先处理出每个数左边比它小 大,右边比它大 小的数目,用cnt[][i]表示.最后统计一下减去重复的就可以 1 #include <iostream> 2 #include <cstdio> 3 #include <cstring> 4 #include <