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

本文介绍了一种用于处理区间内数字标记的问题解决方法。通过在一维数组中对特定位置的数字进行标记,实现对指定区间内首次出现的数字进行计数,并针对多个查询区间求解。该算法使用了离线处理和树状数组来优化查询效率。

题目:

我是超链接

题解:

把第一次在这个区间内出现的所有数字标1,求区间和......

在第一次出现这个数的地方标1,当你的左边(d[i].l)路过的时候,你的1应该消失并去到下一个这样的数字上-----基本思路

(或者:当你的右边(d[i].r)路过的时候,你的1应该出现并让上一个这样的数消失

代码:

#include <cstdio>
#include <algorithm>
#include <cstring>
using namespace std;
int c[50005],a[50005],ans[50005],pd[1000005];
int n,m;
struct hh{int l,r,num;}d[50005];
int cmp(hh a,hh b){return a.r<b.r;}
void add(int loc,int value)
{
	for (int i=loc;i<=n;i+=i&(-i))
	  c[i]+=value;
}
int qurry(int loc)
{
	int ans=0;
	for (int i=loc;i>0;i-=i&(-i))
	  ans+=c[i];
	return ans;
}
int main()
{
	int i,j;
	scanf("%d",&n);
	for (i=1;i<=n;i++)
	  scanf("%d",&a[i]);
	scanf("%d",&m);
	for (i=1;i<=m;i++)
	{
		scanf("%d%d",&d[i].l,&d[i].r);
		d[i].num=i;
	}  
	sort(d+1,d+m+1,cmp);
	int l=0,r=1,ll,k=1;
 	for (i=1;i<=n;i++)
	{		
	    if (!pd[a[i]])
	    {
	    	pd[a[i]]=i;
	    	add(i,1);
		}
		else
		{
			add(pd[a[i]],-1);
			add(i,1); pd[a[i]]=i;
		}
		while (d[k].r==i)
		{
			ans[d[k].num]=qurry(d[k].r)-qurry(d[k].l-1);
			k++;
		}
	}
    for (i=1;i<=m;i++)
      printf("%d\n",ans[i]);
}


# 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、付费专栏及课程。

余额充值