【SDOI2009】HH的项链(树状数组)

本文介绍了一种利用树状数组解决区间查询问题的方法,通过离线处理和优化,实现了对数组中特定区间的高效求解。文章详细解释了如何避免重复计算,以及如何维护每个元素的最新状态,确保查询的准确性。

一开始想到了之前做的这道题

先来看个例子 

1 3 3 3 5

若要询问[3,5],直接用[1,5]-[1,2]得到的答案是1,这显然是错误的

我们观察问题发生在哪里

[1,2]里的3,受到[3,5]里的3的影响,实际上是不应该算入贡献的,也就是说应该把下标为2的地方减去1,也就是说我们需要维护每个数上次出现的地方,之后进行减一

不过我一直卡在离线过后,枚举询问,但缺不好确定这个地方怎么用代码实现

看了一个神仙的代码才领悟,只需要枚举数组里的数即可,并且由于把离线排过序,再维护一个在询问里的指针即可

#include<bits/stdc++.h>
using namespace std;
const int N=500005;
using namespace std;
template<class T>
inline void read(T &x)
{
	x=0; int f=1;
	static char ch=getchar();
	while(!isdigit(ch)) {if(ch=='-') f=-1;ch=getchar();}
	while(isdigit(ch)) x=x*10+ch-'0',ch=getchar();
	x*=f; 
}
struct Query
{
	int l,r,id;
	bool operator <(const Query &that) const
	{
		if(this->r==that.r)	return this->l<that.l;
		return this->r<that.r;
	}
}q[N];
int n,m,ans[N],a[N],tree[N],last[N];
inline int lowbit(int x){return x&(-x);} 
inline void add(int x,int del){for(;x<=n;x+=lowbit(x)) tree[x]+=del;}
inline int query(int x){int ans=0; for(;x;x-=lowbit(x)) ans+=tree[x]; return ans;} 
int main()
{
	ios::sync_with_stdio(false);
	cin.tie(NULL); cout.tie(NULL);
	read(n);
	for(int i=1;i<=n;i++)	read(a[i]);	
	read(m);
	for(int i=1;i<=m;i++) read(q[i].l),read(q[i].r),q[i].id=i;
	sort(q+1,q+m+1);
	int p=1;
	for(int i=1;i<=n;i++)
	{
		if(last[a[i]]) add(last[a[i]],-1);
		add(i,1); last[a[i]]=i;
		while(q[p].r==i) ans[q[p].id]=query(q[p].r)-query(q[p].l-1),p++;
	}
	for(int i=1;i<=m;i++)	cout<<ans[i]<<'\n';
	return 0;
}

 

# P1972 [SDOI2009] HH项链 ## 题目描述 HH 有一串由各种漂亮的贝壳组成的项链HH 相信不同的贝壳会带来好运,所以每次散步完后,他都会随意取出一段贝壳,思考它们所表达的含义。HH 不断地收集新的贝壳,因此,他的项链变得越来越长。 有一天,他突然提出了一个问题:某一段贝壳中,包含了多少种不同的贝壳?这个问题很难回答…… 因为项链实在是太长了。于是,他只好求助睿智的你,来解决这个问题。 ## 输入格式 一行一个正整数 $n$,表示项链长度。 第二行 $n$ 个正整数 $a_i$,表示项链中第 $i$ 个贝壳的种类。 第三行一个整数 $m$,表示 HH 询问的个数。 接下来 $m$ 行,每行两个整数 $l,r$,表示询问的区间。 ## 输出格式 输出 $m$ 行,每行一个整数,依次表示询问对应的答案。 ## 输入输出样例 #1 ### 输入 #1 ``` 6 1 2 3 4 3 5 3 1 2 3 5 2 6 ``` ### 输出 #1 ``` 2 2 4 ``` ## 说明/提示 【数据范围】 对于 $20\%$ 的数据,$1\le n,m\leq 5000$; 对于 $40\%$ 的数据,$1\le n,m\leq 10^5$; 对于 $60\%$ 的数据,$1\le n,m\leq 5\times 10^5$; 对于 $100\%$ 的数据,$1\le n,m,a_i \leq 10^6$,$1\le l \le r \le n$。 本题可能需要较快的读入方式,最大数据点读入数据约 20MB#include <bits/stdc++.h> #define p pair<int,int> #define lowbit(x) x&-x using namespace std; const int N = 1e6+5, M = 1e6+5; int n, m, now = 1;//now为查询双指针 int tree[N], ans[M];//树状数组、答案数组 p arr[N];//元素存储数组,[value,pos] void update(int pos) {//自pos往后加一 for (; pos <= n; pos += lowbit(pos)) tree[pos]++; } int query(int pos) {//自pos往前累加 int res = 0; for (; pos >= 1; pos -= lowbit(pos)) res += tree[pos]; return res; } struct qs { int L, R, X, pos;//左右区间,查询值,查询顺序 } q[M]; int main() { ios::sync_with_stdio(0); cin.tie(0), cout.tie(0); cin >> n >> m; for (int i = 1; i <= n; i++) { cin >> arr[i].first; arr[i].second = i; } sort(arr + 1, arr + 1 + n, [](const p & a, const p & b) {//按值降序 return a.first > b.first; }); for (int i = 1; i <= m; i++) { cin >> q[i].L >> q[i].R >> q[i].X; q[i].pos = i; } sort(q + 1, q + 1 + m, [](const qs & a, const qs & b) {//按值降序 return a.X > b.X; }); for (int i = 1; i <= m; i++) { while (arr[now].first >= q[i].X && now <= n) {//now.value>=query.x则更新 update(arr[now].second); now++; } ans[q[i].pos] = query(q[i].R) - query(q[i].L - 1);//区间查询 } for (int i = 1; i <= m; i++)//输出答案 cout << ans[i] << '\n'; return 0; } 之前的题目我已经用上面这个代码通过了,我想要知道,离线树状数组如何处理种类查询问题
最新发布
08-17
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值