[主席树]HDOJ3874 Necklace

时间:2023-03-09 07:44:41
[主席树]HDOJ3874 Necklace

题意:n个数 m个询问

询问的是[l, r]区间内不同的数的和

没有修改,静态的主席树即可

SPOJ QUERY 一样 将重复的元素建树即可

注意范围:$N \le  50000$ 每个值不超过1000000

也就是加起来会爆int 要用LL

 #include <bits/stdc++.h>
using namespace std;
typedef long long LL;
#define lson l, m
#define rson m+1, r
const int N=5e4+; int L[N<<], R[N<<];
LL sum[N<<];
int tot;
int a[N], T[N];
int read()
{
char ch=' ';
int ans=;
while(ch<'' || ch>'')
ch=getchar();
while(ch<='' && ch>='')
{
ans=ans*+ch-'';
ch=getchar();
}
return ans;
} int build(int l, int r)
{
int rt=(++tot);
sum[rt]=;
if(l<r)
{
int m=(l+r)>>;
L[rt]=build(lson);
R[rt]=build(rson);
}
return rt;
} int update(int pre, int l, int r, int x, int val)
{
int rt=(++tot);
L[rt]=L[pre], R[rt]=R[pre], sum[rt]=sum[pre]+val;
if(l<r)
{
int m=(l+r)>>;
if(x<=m)
L[rt]=update(L[pre], lson, x, val);
else
R[rt]=update(R[pre], rson, x, val);
}
return rt;
} LL query(int u, int v, int l, int r, int k)
{
if(l>=k)
return sum[v]-sum[u];
int m=(l+r)>>;
LL ans=;
if(m>=k)
ans+=query(L[u], L[v], lson, k);
ans+=query(R[u], R[v], rson, k);
return ans;
} LL all[N];
int main()
{
int t;
scanf("%d", &t);
while(t--)
{
tot=;
int n=read();
// scanf("%d", &n);
all[]=;
for(int i=; i<=n; i++)
{
// scanf("%d", &a[i]);
a[i]=read();
all[i]=all[i-]+a[i];
}
T[]=;
map<int, int> mp;
mp.clear();
for(int i=; i<=n; i++)
{
if(mp.find(a[i])!=mp.end())
T[i]=update(T[i-], , n, mp[a[i]], a[i]);
else
T[i]=T[i-];
mp[a[i]]=i;
}
int m=read();
// scanf("%d", &m);
while(m--)
{
int l, r;
l=read(), r=read();
// scanf("%d%d", &l, &r);
printf("%I64d\n", all[r]-all[l-]-query(T[l-], T[r], , n, l));
}
}
return ;
}

HDOJ 3874