[bzoj2743][HEOI2012]采花(树状数组+离线)

2743: [HEOI2012]采花

Time Limit: 15 Sec  Memory Limit: 128 MB
Submit: 1832  Solved: 954
[Submit][Status][Discuss]

Description

萧芸斓是Z国的公主,平时的一大爱好是采花。

今天天气晴朗,阳光明媚,公主清晨便去了皇宫中新建的花园采花。花园足够大,容纳了n朵花,花有c种颜色(用整数1-c表示),且花是排成一排的,以便于公主采花。公主每次采花后会统计采到的花的颜色数,颜色数越多她会越高兴!同时,她有一癖好,她不允许最后自己采到的花中,某一颜色的花只有一朵。为此,公主每采一朵花,要么此前已采到此颜色的花,要么有相当正确的直觉告诉她,她必能再次采到此颜色的花。由于时间关系,公主只能走过花园连续的一段进行采花,便让女仆福涵洁安排行程。福涵洁综合各种因素拟定了m个行程,然后一一向你询问公主能采到多少朵花(她知道你是编程高手,定能快速给出答案!),最后会选择令公主最高兴的行程(为了拿到更多奖金!)。

Input

第一行四个空格隔开的整数n、c以及m。接下来一行n个空格隔开的整数,每个数在[1, c]间,第i个数表示第i朵花的颜色。接下来m行每行两个空格隔开的整数l和r(l ≤ r),表示女仆安排的行程为公主经过第l到第r朵花进行采花。

Output

共m行,每行一个整数,第i个数表示公主在女仆的第i个行程中能采到的花的颜色数。

Sample Input

5 3 5
1 2 2 3 1
1 5
1 2
2 2
2 3
3 5

Sample Output

2
0 0 1 0
【样例说明】
询问[1, 5]:公主采颜色为1和2的花,由于颜色3的花只有一朵,公主不采;询问[1, 2]:颜色1和颜色2的花均只有一朵,公主不采;
询问[2, 2]:颜色2的花只有一朵,公主不采;
询问[2, 3]:由于颜色2的花有两朵,公主采颜色2的花;
询问[3, 5]:颜色1、2、3的花各一朵,公主不采。

HINT

【数据范围】

对于100%的数据,1 ≤ n ≤    10^6,c ≤ n,m ≤10^6。

Source

这道题乍一看很像bzoj1878

但其实要更难

(请先参阅bzoj1878  http://www.cnblogs.com/Pumbit-Legion/p/5874133.html)

上一道题其实是弱化版本

“求区间内一个及以上的颜色数量”

而这道题是“两个及以上”

如何变形是一个问题

对于每个要加入的元素,可以判断它下一个元素是否存在,存在再加一

同理,要出去的话,先把因为这个元素而加入的元素删掉,再看看删掉的能不能带来新的元素

因此,这个问题还能推广到3次,4次等

 1 #include<stdio.h>
 2 #include<stdlib.h>
 3 #include<string.h>
 4 #include<algorithm>
 5 #define LL long long
 6 using namespace std;
 7 typedef struct{
 8     int l,r;
 9     int pos;
10 }qry;
11 qry qr[1000010];
12 int a[1000050];
13 int now[1001000],next[1000050];
14 int bit[1000050],n;
15 int ans[1000050];
16 int cmp1(const qry a,const qry b){
17     if(a.l<b.l)return 1;
18     else if(a.l==b.l)return a.r<b.r;
19     else return 0;
20 }
21 int lb(int x){
22     return x&(-x);
23 }
24 int c1(int x){
25     while(x<=n){
26         bit[x]++;
27         x+=lb(x);
28     }
29     return 0;
30 }
31 int c2(int x){
32     while(x<=n){
33         bit[x]--;
34         x+=lb(x);
35     }
36     return 0;
37 }
38 int q(int x){
39     int ans=0;
40     while(x){
41         ans+=bit[x];
42         x-=lb(x);
43     }
44     return ans;
45 }
46 int main(){
47     int m,c;
48     scanf("%d %d %d",&n,&c,&m);
49     for(int i=1;i<=n;i++){
50         scanf("%d",&a[i]);
51     }
52     for(int i=n;i>0;i--){
53         next[i]=now[a[i]];
54         now[a[i]]=i;
55     }
56     for(int i=1;i<=c;i++)if(next[now[i]])c1(next[now[i]]);
57     for(int i=1;i<=m;i++){
58         scanf("%d %d",&qr[i].l,&qr[i].r);
59         qr[i].pos=i;
60     }
61     sort(qr+1,qr+m+1,cmp1);
62     int l=1;
63     for(int i=1;i<=m;i++){
64         //?????
65         while(l<qr[i].l){
66             if(next[l])c2(next[l]);
67             if(next[next[l]])c1(next[next[l]]);
68             l++;
69         }
70         ans[qr[i].pos]=q(qr[i].r)-q(qr[i].l-1);
71     }
72     for(int i=1;i<=m;i++)printf("%d\n",ans[i]);
73     return 0;
74 }

时间: 2024-12-11 16:42:47

[bzoj2743][HEOI2012]采花(树状数组+离线)的相关文章

[BZOJ2743] [HEOI2012] 采花 (树状数组)

Description 萧芸斓是Z国的公主,平时的一大爱好是采花. 今天天气晴朗,阳光明媚,公主清晨便去了皇宫中新建的花园采花.花园足够大,容纳了n朵花,花有c种颜色(用整数1-c表示),且花是排成一排的,以便于公主采花.公主每次采花后会统计采到的花的颜色数,颜色数越多她会越高兴!同时,她有一癖好,她不允许最后自己采到的花中,某一颜色的花只有一朵.为此,公主每采一朵花,要么此前已采到此颜色的花,要么有相当正确的直觉告诉她,她必能再次采到此颜色的花.由于时间关系,公主只能走过花园连续的一段进行采花

[HEOI2012]采花 树状数组

题目描述 萧薰儿是古国的公主,平时的一大爱好是采花. 今天天气晴朗,阳光明媚,公主清晨便去了皇宫中新建的花园采花. 花园足够大,容纳了n朵花,花有c种颜色(用整数1-c表示),且花是排成一排的,以便于公主采花.公主每次采花后会统计采到的花的颜色数,颜色数越多她会越高兴!同时,她有一癖好,她不允许最后自己采到的花中,某一颜色的花只有一朵.为此,公主每采一朵花,要么此前已采到此颜色的花,要么有相当正确的直觉告诉她,她必能再次采到此颜色的花. 由于时间关系,公主只能走过花园连续的一段进行采花,便让女仆

Luogu4113 HEOI2012 采花 树状数组

传送门 发现这种重复只算一个的题目老是想不出来-- 老套路,先对询问离线,按照右端点从小到大排序.然后我们考虑在右端点右移时新加入的那些位置的贡献,设$pre_i$表示与第$i$个位置相同数字的最后一个位置,那么将第$t$个数字放入的时候,$pre_t$就会产生$1$的贡献,而$pre_{pre_t}$的贡献因为$pre_t$的贡献而变成$0$,可以使用树状数组维护,每一次查询一段区间的贡献和即可. 1 #include<bits/stdc++.h> 2 //This code is writ

BZOJ 2743 HEOI2012 采花 树状数组

题目大意:给定一个序列,多次询问区间内出现两次以上的数的数量 n<=100W 莫队不用想了 考虑对于每个区间的左端点 对这个区间有贡献的数是从这个端点开始所有第二次出现的数 于是我们将区间按照左端点排序  然后从左向右扫 令next[i]为i位置上的数下一次出现的位置 初始将所有第二次出现的数加入树状数组 然后每删除一个点i 将next[i]从树状数组中删除 然后将next[next[i]]加入树状数组 然后处理区间就在树状数组上查询右端点即可 #include<cstdio> #inc

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(╯

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

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

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

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

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

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