浅谈树状数组与线段树:https://www.cnblogs.com/AKMer/p/9946944.html

题目传送门:https://www.lydsy.com/JudgeOnline/problem.php?id=2120

对于区间内颜色种类数有一个百用不倦的套路,那就是只有每种颜色只有最左/右边算数,其它都算没颜色。

这一题也是一样,对于每个位置,我记录一下\(nxt[i]\)表示下一个离\(i\)最近的颜色与\(i\)相同的位置在哪。问题就变成了询问在区间\([l,r]\)内,有多少位置的\(nxt\)是大于\(r\)的了。用二维线段树就轻松解决了。

时间复杂度:\(O(nlog^2n)\)

空间复杂度:\(O(nlog^2n)\)

代码如下:

#include <set>
#include <cstdio>
#include <iostream>
#include <algorithm>
using namespace std; const int maxn=5e4+5,maxv=1e6+5; int n,m;
char ch[5];
int a[maxn];
set<int>s[maxv];
set<int>::iterator it;
int lst[maxn],nxt[maxn]; int read() {
int x=0,f=1;char ch=getchar();
for(;ch<'0'||ch>'9';ch=getchar())if(ch=='-')f=-1;
for(;ch>='0'&&ch<='9';ch=getchar())x=x*10+ch-'0';
return x*f;
} struct nxt_segment_tree {
int tot;
int sum[maxn*100],ls[maxn*100],rs[maxn*100]; void change(int p,int l,int r,int pos,int v) {
while(1) {
sum[p]+=v;
if(l==r)break;
int mid=(l+r)>>1;
if(pos<=mid) {
if(!ls[p])ls[p]=++tot;
p=ls[p],r=mid;
}
else {
if(!rs[p])rs[p]=++tot;
p=rs[p],l=mid+1;
}
}
} int query(int p,int l,int r,int pos) {
int res=0;
while(l!=r) {
int mid=(l+r)>>1;
if(pos<=mid) {
res+=sum[rs[p]];
r=mid,p=ls[p];
}
else l=mid+1,p=rs[p];
}
return res;
}
}T_inside; struct pos_segment_tree {
int rt[maxn<<2]; void change(int p,int l,int r,int pos,int v,int opt) {
while(1) {
if(!rt[p])rt[p]=++T_inside.tot;
T_inside.change(rt[p],0,n+1,v,opt);
if(l==r)return;int mid=(l+r)>>1;
if(pos<=mid)r=mid,p=p<<1;
else l=mid+1,p=p<<1|1;
}
} int query(int p,int l,int r,int L,int R,int v) {
if(L<=l&&r<=R)return T_inside.query(rt[p],0,n+1,v);
int mid=(l+r)>>1,res=0;
if(L<=mid)res+=query(p<<1,l,mid,L,R,v);
if(R>mid)res+=query(p<<1|1,mid+1,r,L,R,v);
return res;
}
}T_outside; int main() {
n=read(),m=read();
for(int i=1;i<=n;i++)
a[i]=read(),s[a[i]].insert(i);
for(int i=1;i<=n;i++)
if(!s[a[i]].empty()) {
s[a[i]].insert(0);
s[a[i]].insert(n+1);
}
for(int i=1;i<=1000000;i++)
for(it=s[i].begin();it!=s[i].end();it++) {
int pos=*it;
if(pos==0||pos==n+1)continue;
if(pos!=0) {it--,lst[pos]=*it,it++;}
if(pos!=n+1) {it++,nxt[pos]=*it,it--;}
T_outside.change(1,1,n,pos,nxt[pos],1);
}
for(int i=1;i<=m;i++) {
scanf("%s",ch+1);
if(ch[1]=='Q') {
int l=read(),r=read();
printf("%d\n",T_outside.query(1,1,n,l,r,r));
}
else {
int pos=read(),col=read();
if(s[col].empty())s[col].insert(0),s[col].insert(n+1);
int x=lst[pos],y=nxt[pos];
if(x)T_outside.change(1,1,n,x,pos,-1),T_outside.change(1,1,n,x,y,1);
T_outside.change(1,1,n,pos,y,-1);nxt[x]=y,lst[y]=x;
it=s[a[pos]].find(pos),s[a[pos]].erase(it);
it=s[col].lower_bound(pos),y=*it,it--,x=*it,s[col].insert(pos);
if(x)T_outside.change(1,1,n,x,y,-1),T_outside.change(1,1,n,x,pos,1);
T_outside.change(1,1,n,pos,y,1);
nxt[x]=pos,nxt[pos]=y,lst[y]=pos,lst[pos]=x;
a[pos]=col;
}
}
return 0;
}

BZOJ2120:数颜色的更多相关文章

  1. BZOJ2120 数颜色 【带修莫队】

    BZOJ2120 数颜色 Description 墨墨购买了一套N支彩色画笔(其中有些颜色可能相同),摆成一排,你需要回答墨墨的提问.墨墨会像你发布如下指令: 1. Q L R代表询问你从第L支画笔到 ...

  2. bzoj2120 数颜色 莫队 带修改

    [bzoj2120]数颜色 Description 墨墨购买了一套N支彩色画笔(其中有些颜色可能相同),摆成一排,你需要回答墨墨的提问.墨墨会像你发布如下指令: 1. Q L R代表询问你从第L支画笔 ...

  3. BZOJ2120 数颜色 【带修改莫队】

    2120: 数颜色 Time Limit: 6 Sec  Memory Limit: 259 MB Submit: 6579  Solved: 2625 [Submit][Status][Discus ...

  4. BZOJ2120 数颜色 —— 待修改莫队

    题目链接:https://vjudge.net/problem/HYSBZ-2120 2120: 数颜色 Time Limit: 6 Sec  Memory Limit: 259 MBSubmit:  ...

  5. BZOJ2120 数颜色(树套树)

    B. 数颜色 题目描述 墨墨购买了一套N支彩色画笔(其中有些颜色可能相同),摆成一排,你需要回答墨墨的提问.墨墨会像你发布如下指令:1. Q L R代表询问你从第L支画笔到第R支画笔中共有几种不同颜色 ...

  6. [Bzoj2120]数颜色 (非正解 )(莫队)

    2120: 数颜色 Time Limit: 6 Sec  Memory Limit: 259 MBSubmit: 6286  Solved: 2489[Submit][Status][Discuss] ...

  7. BZOJ2453维护队列&&BZOJ2120数颜色

    2016-05-28 11:20:22 共同的思路: 维护某种颜色上一次在哪里出现pre,可以知道当pre<询问的l时更新答案 块内按照pre排序 修改的时候重新O(n)扫一遍,如果和之前的不一 ...

  8. bzoj2120: 数颜色(BIT套主席树+set/分块)

    带修改的 HH的项链. 带修改考虑用BIT套主席树,查区间里有几个不同的数用a[i]上次出现的位置pre[i]<l的数有几个来算就好了. 考虑怎么修改.修改i的时候,我们需要改变i同颜色的后继的 ...

  9. BZOJ2120 数颜色(带修改莫队)

    本文版权归ljh2000和博客园共有,欢迎转载,但须保留此声明,并给出原文链接,谢谢合作. 本文作者:ljh2000作者博客:http://www.cnblogs.com/ljh2000-jump/转 ...

  10. BZOJ2120 数颜色

    Description 墨墨购买了一套N支彩色画笔(其中有些颜色可能相同),摆成 一排,你需要回答墨墨的提问.墨墨会像你发布如下指令: 1. Q L R代表询问你从第L支画笔到第R支画笔中共有几种不同 ...

随机推荐

  1. ORACLE client 11g r2 客户端开发环境配置

    一.安装ORACLE客户端,这里不做说明.需要注意的是,客户端解压位置应该在磁盘根目录下. 如果放在带中文字或者空格的文件名的路径下出了问题,可以放到磁盘根目录在安装.应该就会没有问题. 另外,一般安 ...

  2. 【题解】P2161[SHOI2009]会场预约(set)

    [题解][P2161 SHOI2009]会场预约 题目很像[[题解]APIO2009]会议中心 \(set\)大法好啊! 然后我们有个小\(trick\)(炒鸡帅),就是如何优雅地判断线段交? str ...

  3. linux c编程:文件夹操作

    创建目录: 用mkdir函数创建目录: mkdir(const char *pathname, mode_t mode) 参数mode有下列的组合: S_ISUID 04000 文件的执行时设置用户I ...

  4. Call method 的使用

    SAP学习日志---Call method 的使用 以及常见错误 转载▼   可以通过以下方法 call method 1. 进入全局类中 找到方法,拖到程序中 2. 使用pattern 中的 AAB ...

  5. Django中的Cookie和Session操作以及CBV

    1.Cookie 平常我们在浏览网页的时候,在需要输入密码的地方,如果已经登陆了一次,并且时间间隔比较近的话,是不需要登陆的,为什么了?这就是Cookie的作用. Cookie(或Cookies)指某 ...

  6. 微信小程序高度设置为100%

    在网页中设置body,html{height:100%}; 将body和html设置为100%,这样我们就可以在他们的子元素中使用height:100%来使的我们的容器元素占满屏幕的高度啦. 但是在微 ...

  7. Kattis - fairdivision 【贪心】

    题意 有一堆人 要给他们的朋友 买一个生日礼物,然后 每个人 给出自己的最大负担额度 并且给出礼物总价 然后要给出一种解决方案 尽量让 所有人的支出都接近平均,如果实在无法平均,那就让 先来的人 多处 ...

  8. iOS 当前应用或者浏览器中 唤起 手机其他应用

    这种方法 是 产品很常见的需求,关键 是在info.plist  URL types 设置对应属性 比如 里面 子属性 URL identifier  设置成 bundle id   //设置应用指向 ...

  9. Python OOP(1)

    类(Class): 用来描述具有相同的属性和方法的对象的集合.它定义了该集合中每个对象所共有的属性和方法.对象是类的实例. 类变量:类变量在整个实例化的对象中是公用的.类变量定义在类中且在函数体之外. ...

  10. JVM - 堆外内存

    看了不少资料,总结下: 堆外内存 / 直接内存(Direct Memory)JDK1.4中引入的NIO类,基于channel和Buffer的I/O方式,可用Native库直接分配堆外内存,然后利用一个 ...