题目

有n个数编号从0→n-1,两种操作:

Q L R:询问编号为L→R-1的数中共有多少种不同的数

M X Y:将编号为X的数改为Y

共有m个操作

分析

既然是单点修改,查询,我们考虑一下分块。

首先,定义\(next_{i}\)表示,在\(i\)之后的第一个与编号为\(i\)的数相同的数的位置。

接着,我们把\(i\)向\(next_{i}\)连一条边。

那么就会发现,当把边处理好后,查询操作就迎刃而解了;

查询操作

假设现在要查询\([x,y]\),



其中next,以及它们的连边的情况是:



发现,编号为\(a\)的数和编号为\(next_a\)的数的数以及编号为\(b\)的数和编号为\(next_b\)的数的数都重复了。

那么就要减去重复的,

而编号为\(next_c\)的数并不在\([x,y]\),所以并不需要减去。

我们得出一个结论:\(\color{red}{当next_i<=y时,编号为i的数一定有重复,所以我们就只记录最后一个,即next_i>y的那一个}\)

那如何处理呢?

对于不是整块的部分,暴力处理。时间复杂度\(O(2\sqrt{n})\)。

对于整块的部分,把整块按\(next_{i}\)排序,二分求答案。

修改操作

我们再定义\(last_{i}\)表示,在\(i\)之前的第一个与编号为\(i\)的数相同的数的位置。

再把\(i\)向\(last_{i}\)连一条边;

假设我们修改位置x,



那么,因为x修改成别的值,所以\(next[last[x]]\)就要就修成\(next[x]\),\(last[next[x]]\)就要就改成\(last[x]\)

接着,发现,这样\(next[last[x]]\)改变了,于是,重新把\(last[x]\)所在的块排序。

然后,就要处理位置x新的\(last和next\)。

假设x所在的块的开头和结尾是s和t。



处理\(last\):

我们先暴力查看x-1~s的位置中是否出现过y,如果有,修改\(last[x]\)以及\(next[last[x]]\)。

否则查看前面的块,

再定义\(sum[i][j]\)表示,在第i个块中,j这个数出现过多少次。

找到了,就修改就可以了。

处理\(next\)类似。

记住,如果某个位置的\(next\)修改了,就将这个位置所在的块排个序。

#include <cmath>
#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <algorithm>
#include <queue>
const int maxlongint=2147483647;
const int mo=1000000007;
const int N=50005;
using namespace std;
int a[N*2],n,m,ans,next[N*2],last[N*2],part[N][2],size,as[N*2],pos[N*2],stead[1000005],sum[250][N*2],tot,color[N*2],num;
bool cmp(int x,int y)
{
return next[x]<next[y];
}
int preblock()
{
size=sqrt(n);
for(int i=1;i<=n;i+=size)
{
part[++tot][0]=i;
if(i+size-1>n)
part[tot][1]=n;
else
part[tot][1]=i+size-1;
}
for(int i=1;i<=tot;i++)
for(int j=part[i][0];j<=part[i][1];j++)
{
sum[i][stead[a[j]]]++;
pos[j]=i;
}
}
int prenexus()
{
memset(color,0,sizeof(color));
for(int i=0;i<=n;i++)
{
last[i]=color[stead[a[i]]];
color[stead[a[i]]]=i;
}
memset(color,0,sizeof(color));
for(int i=n;i>=0;i--)
{
next[i]=color[stead[a[i]]];
if(!next[i])
next[i]=maxlongint;
color[stead[a[i]]]=i;
}
}
int so(int x)
{
sort(as+part[x][0],as+part[x][1]+1,cmp);
}
int fs(int x,int y,int y1)
{
for(int i=x;i<=y;i++)
if(next[i]>y1)
ans++;
}
int zt(int x,int y,int y1)
{
int ll=x-1,rr=y;
while(ll<rr-1)
{
int mid=(ll+rr)/2;
if(next[as[mid]]<=y1) ll=mid;
else rr=mid;
}
int q=0;
if(next[as[rr]]<=y1) q=rr;
else q=ll;
ans+=y-x+1-(q-x+1);
}
int main()
{
scanf("%d%d",&n,&m);
for(int i=1;i<=n;i++)
{
scanf("%d",&a[i]);
if(!stead[a[i]])
{
stead[a[i]]=++num;
}
}
preblock();
prenexus();
for(int i=1;i<=n;i++) as[i]=i;
for(int i=1;i<=tot;i++) so(i);
for(int k=1;k<=m;k++)
{
char c=getchar();
while(c!='Q' && c!='M') c=getchar();
int x,y;
scanf(" %d %d",&x,&y);
x+=1;
if(c=='Q')
{
ans=0;
int l=pos[x],r=pos[y];
if(l==r)
{
for(int i=x;i<=y;i++)
if(next[i]>y) ans++;
printf("%d\n",ans);
continue;
}
if(x>part[l][0])
{
for(int i=x;i<=part[l][1];i++)
if(next[i]>y) ans++;
l++;
}
if(y<part[r][1])
{
for(int i=part[r][0];i<=y;i++)
if(next[i]>y) ans++;
r--;
}
for(int i=l;i<=r;i++)
zt(part[i][0],part[i][1],y);
printf("%d\n",ans);
}
else
{
bool q=true;
if(!stead[y])
{
stead[y]=++num;
q=false;
}
next[last[x]]=next[x];
if(next[x]!=maxlongint)
last[next[x]]=last[x];
sum[pos[x]][stead[a[x]]]--;
sum[pos[x]][stead[y]]++;
so(pos[last[x]]);
a[x]=y;
if(!q)
{
last[x]=0;
next[x]=maxlongint;
}
else
{
int p=0,p1=0;
for(int i=x-1;i>=part[pos[x]][0];i--)
if(a[i]==y)
{
p=i;
break;
}
if(!p)
{
for(int i=pos[x]-1;i>=1;i--)
{
if(sum[i][stead[y]])
{
for(int j=part[i][1];j>=part[i][0];j--)
if(a[j]==y)
{
p=j;
break;
}
break;
}
}
}
if(p)
{
last[x]=p;
next[p]=x;
so(pos[p]);
}
else
last[x]=0;
for(int i=x+1;i<=part[pos[x]][1];i++)
if(a[i]==y)
{
p1=i;
break;
}
if(!p1)
{
for(int i=pos[x]+1;i<=tot;i++)
{
if(sum[i][stead[y]])
{
for(int j=part[i][0];j<=part[i][1];j++)
if(a[j]==y)
{
p1=j;
break;
}
break;
}
}
}
if(p1)
{
next[x]=p1;
last[p1]=x;
so(pos[x]);
}
else
next[x]=maxlongint;
}
so(pos[x]);
}
}
}

Dynamic len的更多相关文章

  1. Uva 3767 Dynamic len(set(a[L:R])) 树套树

    Dynamic len(set(a[L:R])) Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 https://uva.onlinejudge.org/in ...

  2. UVA 12345 Dynamic len(带修莫队)

    Dynamic len [题目链接]Dynamic len [题目类型]带修莫队 &题解: 莫队可以单点更改,只要再多加一维,代表查询次数,排序的时候3个关键字. 之后循环离线的时候,先暴力时 ...

  3. 【题解】Luogu UVA12345 Dynamic len(set(a[L:R]))

    原题传送门 这题要用动态莫队,我博客里有介绍 这道题和luogu P1903 [国家集训队]数颜色 / 维护队列差不多,解法就在上面那篇博客里qaq 主要的问题是如何排序? 排序有三个关键字: 1.左 ...

  4. (待修莫队 没过! 抽空在检查)Dynamic len(set(a[L:R])) UVA - 12345

    #include <iostream> #include <cstdio> #include <sstream> #include <cstring> ...

  5. Dynamic len(set(a[L:R])) UVA - 12345(这么过分一定要写博客)

    给出一个有n个元素的数组,有以下两种操作:Q x y,求出区间[x,y)内不同元素的个数, M x y,把第x个元素的值修改为y.注意题目中的下标是从0开始的 这题超级超级坑 妈的一个水题找了几个小时 ...

  6. UVA 12345 Dynamic len(set(a[LR]))

    题意:询问区间唯一元素个数,单点修改. 分析: 借助Unique snowflakes, Can you answer these queries II的思想,唯一性可以借助元素上一次出现的位置来判断 ...

  7. [UVa12345] Dynamic len (带 修 )

    题意:有n个数编号从0→n-1,两种操作:            Q L R:询问编号为L→R-1的数中共有多少种不同的数             M X Y:将编号为X的数改为Y           ...

  8. UESTC-1259 昊昊爱运动 II

    昊昊爱运动 II Time Limit: 3000/1000MS (Java/Others)     Memory Limit: 65535/65535KB (Java/Others)     昊昊喜 ...

  9. UVA12345 (带修改的莫队)

    UVA12345 Dynamic len Problem : 给一个序列,每次询问一个区间里面的数字种类数量,或者修改某一个位置的值. Solution : 第一关键字分块排序左端点,第二关键字分块排 ...

随机推荐

  1. tuple用法

    1 tuple中的元素可以直接赋给相同个数的变量 tup1 = ('asfa',234) p, q = tup1 print(p) print(q) # asfa # 参考:https://www.r ...

  2. 【Fiddler】开启手机的http或https抓包

    fiddler安装 下载fiddler最新版: 默认安装: 打开fiddler工具,默认界面: 选择上方,Tools-→options General界面 HTTPS界面 CONNECTIONS,po ...

  3. windows 的cmd设置代理的问题

    今天给公司一同事用cmd来安装gulp(npm install -g gulp), 死活安装不上,一直报一大堆的错误:经仔细查阅是代理的问题,故总结如下: 若公司的电脑是通过设置代理来访问外网,则需要 ...

  4. Java多线程学习——知识点积累

    开启多线程时,每一个线程都拥有自己的工作空间,每个工作空间都单独的和主存打交道. 并发的概念:多个线程同时操作同一个对象 当产生并发时,如果从工作空间写入数据到内存的线程时间片用完了,其他线程再从主存 ...

  5. 简述Vue项目中返回上一页

    1.背景 由于Vue所生成的项目叫做单页应用,即SPA,如果还是使用jQuery中的go(-)或back()是行不通的,所以,我们使用到了Vue中的编程式导航. 2.基本使用 定义返回按钮: < ...

  6. Flask框架 (四)—— 请求上下文源码分析、g对象、第三方插件(flask_session、flask_script、wtforms)、信号

    Flask框架 (四)—— 请求上下文源码分析.g对象.第三方插件(flask_session.flask_script.wtforms).信号 目录 请求上下文源码分析.g对象.第三方插件(flas ...

  7. linux应用程序启动时加载库错误问题

    ldd text查看依赖库 ln -s /lib64/libpcre.so.0 /usr/local/lib/libpcre.so做软连接

  8. SpringBoot项目快速启动停止脚本

    SpringBoot项目快速启动停止脚本 1.在jar包同级目录下,创建 app.sh #!/bin/bash appName=`ls|grep .jar$` if [ -z $appName ] t ...

  9. realloc ------ 扩大malloc得到的内存空间

    char* p = malloc(1024);char* q = realloc(p,2048); 现在的问题是我们应该如何处理指针 p. 刚开始按照我最直观的理解,如果就是直接将 p = NULL; ...

  10. spring @Value 获取配置文件为 null 常见的几种方式

    第一种方式: xx.properties 属性名称错误,未与@Value("${xxx}") 进行对应 第二种方式: 该类未注入到spring bean容器中 @Component ...