HDU 5919-Sequence II(主席树)


题意:给你个长为n的序列,m次查询,每次查询该区间中所有种类的数在该区间中第一次出现的位置的中位数是多少(比较绕)

题解:主席树模板题,很容易想到其实就是找该区间中有多少种数的变形,因为要找每个数在该区间中第一次出现的位置,所以有个技巧是倒着插入主席树,每插入一个值,将当前数位置的贡献值+1,将之前已经出现过的该数的位置上的贡献值-1即可。

#include<math.h>
#include<stdio.h>
#include<string.h>
#include<iostream>
#include<algorithm>
using namespace std;
typedef long long ll;
#define maxn 8000005
int id,sz,a[maxn],rt[maxn],ans[maxn];
int sum[maxn],ls[maxn],rs[maxn],last[maxn];
void update(int last,int &id,int l,int r,int k,int val)
{
	id=++sz;
	ls[id]=ls[last];
	rs[id]=rs[last];
	sum[id]=sum[last]+val;
	if(l==r)
		return;
	int mid=(l+r)/2;
	if(k<=mid)
		update(ls[last],ls[id],l,mid,k,val);
	else
		update(rs[last],rs[id],mid+1,r,k,val);
}
int query(int id,int l,int r,int L,int R)
{
	if(l>=L && r<=R)
		return sum[id];
	int mid=(l+r)/2,res=0;
	if(L<=mid)
		res+=query(ls[id],l,mid,L,R);
	if(R>mid)
		res+=query(rs[id],mid+1,r,L,R);
	return res;
}
int find(int id,int l,int r,int val)
{
	if(l==r)
		return l;
	int mid=(l+r)/2;
	if(sum[ls[id]]>=val)
		return find(ls[id],l,mid,val);
	else
		return find(rs[id],mid+1,r,val-sum[ls[id]]);
}
int main(void)
{
	int T,n,m,i,cases=0;
	scanf("%d",&T);
	while(T--)
	{
		ans[0]=0;sz=0;
		memset(ls,0,sizeof(ls));
		memset(rs,0,sizeof(rs));
		memset(sum,0,sizeof(sum));
		memset(last,0,sizeof(last));
		scanf("%d%d",&n,&m);
		for(i=1;i<=n;i++)
			scanf("%d",&a[i]);
		rt[n+1]=0;
		for(i=n;i>0;i--)
		{
			int head=0;rt[i]=0;
			update(rt[i+1],head,1,n,i,1);
			if(last[a[i]])
				update(head,rt[i],1,n,last[a[i]],-1);
			else
				rt[i]=head;
			last[a[i]]=i;
		}
		int l,r;
		for(i=1;i<=m;i++)
		{
			scanf("%d%d",&l,&r);
			l=(l+ans[i-1])%n+1;
			r=(r+ans[i-1])%n+1;
			if(l>r) swap(l,r);
			int sum=(query(rt[l],1,n,l,r)+1)/2;
			ans[i]=find(rt[l],1,n,sum);
		}
		printf("Case #%d:",++cases);
		for(i=1;i<=m;i++)
			printf(" %d",ans[i]);
		printf("\n");
	}
	return 0;
}


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值