Codeforces 848C Goodbye Souvenir(CDQ 分治)
2024-09-03 05:00:49
考虑记录每个点的前驱 \(pre_x\),显然答案为 \(\sum\limits_{i=l}^{r} i-pre_i (pre_i \geq l)\)
我们建立一个平面直角坐标系,\(x\) 轴表示下标 \(i\),\(y\) 轴表示前驱 \(pre_i\),点权为 \(i-pre_i\)。
每次询问以 \((l,l)\) 为左下角,\((r,n)\) 为右下角的矩形中所有点的权值和。
至于修改操作,就是撤销上次操作的贡献,加入新的贡献。
至此,我们就把问题转化为单点加,矩形和的问题,再加上有时间轴,三维偏序,故可用 CDQ 分治维护。
/*
Contest: -
Problem: Codeforces 848C
Author: tzc_wk
Time: 2020.7.17
*/
#include <bits/stdc++.h>
using namespace std;
#define fi first
#define se second
#define fz(i,a,b) for(int i=a;i<=b;i++)
#define fd(i,a,b) for(int i=a;i>=b;i--)
#define foreach(it,v) for(__typeof(v.begin()) it=v.begin();it!=v.end();it++)
#define all(a) a.begin(),a.end()
#define fill0(a) memset(a,0,sizeof(a))
#define fill1(a) memset(a,-1,sizeof(a))
#define fillbig(a) memset(a,0x3f,sizeof(a))
#define fillsmall(a) memset(a,0xcf,sizeof(a))
#define y1 y1010101010101
#define y0 y0101010101010
#define int long long
typedef pair<int,int> pii;
inline int read(){
int x=0,neg=1;char c=getchar();
while(!isdigit(c)){
if(c=='-') neg=-1;
c=getchar();
}
while(isdigit(c)) x=x*10+c-'0',c=getchar();
return x*neg;
}
const int INF=1e9;
int n=read(),m=read(),k=0,ans[1000005],a[100005];
struct data{
int t,p2,p1,p3;
} p[1000005],tmp[1000005];
set<int> st[100005];
struct BIT{
int bit[2000005];
inline void add(int x,int v){
x++;
for(int i=x;i<=n+1;i+=(i&(-i))) bit[i]+=v;
}
inline int query(int x){
x++;
int sum=0;
for(int i=x;i;i-=(i&(-i))) sum+=bit[i];
return sum;
}
} t;
inline void merge(int l,int r,int mid){
int p1=l,p2=mid+1;
for(int i=l;i<=r;i++){
if(p1<=mid&&(p2>r||p[p1].p2<p[p2].p2))
tmp[i]=p[p1++];
else
tmp[i]=p[p2++];
}
for(int i=l;i<=r;i++){
p[i]=tmp[i];
}
}
inline void CDQ(int l,int r){
if(l==r) return;
int mid=(l+r)>>1;
CDQ(l,mid);CDQ(mid+1,r);
// printf("%d %d\n",l,r);
int cur=l;
for(int i=mid+1;i<=r;i++){
if(p[i].p3!=-INF) continue;
while(cur<=mid&&p[cur].p2<=p[i].p2){
if(p[cur].p3!=-INF) t.add(p[cur].p1,p[cur].p3);
cur++;
}
ans[p[i].t]+=t.query(n)-t.query(p[i].p1-1);
}
fz(i,l,cur-1) if(p[i].p3!=-INF) t.add(p[i].p1,-p[i].p3);
merge(l,r,mid);
}
signed main(){
fz(i,1,n) a[i]=read();
fz(i,1,n) st[i].insert(0);
fz(i,1,n){
p[++k]={k,i,*st[a[i]].rbegin(),i-*st[a[i]].rbegin()};
st[a[i]].insert(i);
}
fz(i,1,n) st[i].insert(n+1);
fill1(ans);
fz(i,1,m){
int op=read(),x=read(),y=read();
if(op==1){
set<int>::iterator pre=--st[a[x]].lower_bound(x);
set<int>::iterator nxt=st[a[x]].upper_bound(x);
p[++k]={k,*nxt,x,-((*nxt)-x)};
p[++k]={k,x,*pre,-(x-(*pre))};
p[++k]={k,*nxt,*pre,(*nxt)-(*pre)};
st[a[x]].erase(x);
a[x]=y;
pre=--st[a[x]].lower_bound(x);
nxt=st[a[x]].upper_bound(x);
p[++k]={k,*nxt,x,((*nxt)-x)};
p[++k]={k,x,*pre,(x-(*pre))};
p[++k]={k,*nxt,*pre,-((*nxt)-(*pre))};
st[a[x]].insert(x);
}
else{
p[++k]={k,y,x,-INF};
ans[k]=0;
}
}
// fz(i,1,k){
// printf("%d %d %d %d\n",p[i].t,p[i].p1,p[i].p2,p[i].p3);
// }
// puts("-1");
CDQ(1,k);
fz(i,1,k) if(ans[i]!=-1) printf("%lld\n",ans[i]);
return 0;
}
最新文章
- 1.JAVA之GUI编程概述
- Oracle入门基础
- [ADB]ADB(Android Debug Bridge)简介及基础(不包含命令)
- MVC控制器总结
- Oracle 10g Block Change Tracking特性
- Mac下同时安装多个版本的JDK &; Mac 可设置环境变量的位置、查看和添加PATH环境变量
- iOS不得姐项目--推荐关注模块(一个控制器控制两个tableView),数据重复请求的问题,分页数据的加载,上拉下拉刷新(MJRefresh)
- Android源码剖析之Framwork层消息传递(Wms到View)
- iOS 细碎知识整理
- 解决VS2008 开发Windows Mobile 项目生成速度慢的问题
- 多线程&;多进程解析:Python、os、sys、Queue、multiprocessing、threading
- Ubuntu 安装
- Windows 编译器选项 Runtime Library
- C# 函数覆盖总结学习
- this.class.getClassLoader()怎么理解?
- HDU-2176 取(m堆)石子游戏
- codeforces 305E Playing with String
- Java String字符串深入详解
- 在laravel环境下将图片存入MongoDB数据库
- mysql(1)—— 详解一条sql语句的执行过程