莫队
1、普通莫队
SP3267 DQUERY - D-query
题意简明易懂:给你一个长度不大于n≤5×10^5的序列,其中数值都小于等于10^6,有m≤5×10^5次询问,每次询问区间[l,r]中数值个数(也就是去重后数字的个数)。
#include <cstdio> #include <cstring> #include <cmath> #include <algorithm> using namespace std; #define maxn 1010000 #define maxb 1010 int aa[maxn], cnt[maxn], belong[maxn]; int n, m, size, bnum, now, ans[maxn]; struct query { int l, r, id; } q[maxn]; int cmp(query a, query b) { return (belong[a.l] ^ belong[b.l]) ? belong[a.l] < belong[b.l] : ((belong[a.l] & 1) ? a.r < b.r : a.r > b.r); } #define isdigit(x) ((x) >= '0' && (x) <= '9') int read() { int res = 0; char c = getchar(); while(!isdigit(c)) c = getchar(); while(isdigit(c)) res = (res << 1) + (res << 3) + c - 48, c = getchar(); return res; } void printi(int x) { if(x / 10) printi(x / 10); putchar(x % 10 + '0'); } int main() { scanf("%d", &n); size = sqrt(n); bnum = ceil((double)n / size); for(int i = 1; i <= bnum; ++i) for(int j = (i - 1) * size + 1; j <= i * size; ++j) { belong[j] = i; } for(int i = 1; i <= n; ++i) aa[i] = read(); m = read(); for(int i = 1; i <= m; ++i) { q[i].l = read(), q[i].r = read(); q[i].id = i; } sort(q + 1, q + m + 1, cmp); int l = 1, r = 0; for(int i = 1; i <= m; ++i) { int ql = q[i].l, qr = q[i].r; while(l < ql) now -= !--cnt[aa[l++]]; while(l > ql) now += !cnt[aa[--l]]++; while(r < qr) now += !cnt[aa[++r]]++; while(r > qr) now -= !--cnt[aa[r--]]; ans[q[i].id] = now; } for(int i = 1; i <= m; ++i) printi(ans[i]), putchar('\n'); return 0; }
带修莫队
Luogu P1903 [国家集训队]数颜色 / 维护队列
墨墨购买了一套N支彩色画笔(其中有些颜色可能相同),摆成一排,你需要回答墨墨的提问。墨墨会向你发布如下指令:
1、 Q L R代表询问你从第L支画笔到第R支画笔中共有几种不同颜色的画笔。
2、 R P Col把第PP支画笔替换为颜色Col。
#pragma GCC optimize(2) #pragma GCC optimize(3) #pragma GCC optimize("Ofast") #pragma GCC optimize("inline") #include<bits/stdc++.h> using namespace std; const int N=5e5+5; int a[N], cnt[N<<1], ans[N], belong[N]; struct query {int l, r, time, id;} q[N]; struct modify {int pos, color, last;} c[N]; int cntq, cntc, n, m, size, bnum; inline int cmp(query a, query b) { return (belong[a.l] ^ belong[b.l]) ? belong[a.l] < belong[b.l] : ((belong[a.r] ^ belong[b.r]) ? belong[a.r] < belong[b.r] : a.time < b.time); } inline int read() { int res = 0; char c = getchar(); while(!isdigit(c)) c = getchar(); while(isdigit(c)) res = (res << 1) + (res << 3) + (c ^ 48), c = getchar(); return res; } int main() { n = read(), m = read(); size = pow(n, 2.0 / 3.0); bnum = ceil((double)n / size); for (int i = 1; i <= bnum; ++i) for (int j = (i-1)*size+1; j <=i*size; ++j) belong[j]=i; for(int i = 1; i <= n; ++i) a[i] = read(); for(int i = 1; i <= m; ++i) { char op; cin>>op; if (op == 'Q') { q[++cntq].l = read(); q[cntq].r = read(); q[cntq].time = cntc; q[cntq].id = cntq; } else if(op == 'R') { c[++cntc].pos = read(); c[cntc].color = read(); } } sort(q + 1, q + cntq + 1, cmp); int l = 1, r = 0, time = 0, now = 0; for(int i = 1; i <= cntq; ++i) { int ql = q[i].l, qr = q[i].r, qt = q[i].time; while (l < ql) now -= !--cnt[a[l++]]; while (l > ql) now += !cnt[a[--l]]++; while (r < qr) now += !cnt[a[++r]]++; while (r > qr) now -= !--cnt[a[r--]]; while (time < qt) { ++time; if (ql <= c[time].pos && c[time].pos <= qr) now-= !--cnt[a[c[time].pos]]-!cnt[c[time].color]++; swap(a[c[time].pos],c[time].color); } while (time > qt) { if (ql <= c[time].pos && c[time].pos <= qr) now-= !--cnt[a[c[time].pos]]-!cnt[c[time].color]++; swap(a[c[time].pos], c[time].color); --time; } ans[q[i].id] = now; } for (int i = 1; i <= cntq; ++i) printf("%d\n", ans[i]); return 0; }
cf940f
给出一个长度为N序列
需要支持两种操作:
1、定义一个区间的值为:这段区间任意元素出现次数的集合的mex,给出l,r求原串中[l,r]这段区间的值
2、修改某个点的值
对mex的定义与SG函数中是相同的,表示一个自然数集中未出现的最小的整数。
例如:1、3、2、1、2、2、2这个序列的值为3:
3出现了1次,1出现了2次,2出现了4次,集合中的数为0,1,2,4(所有除了这三个数以外的[0,10^9]的数均未出现,所以为0)
N,M(操作次数)≤10^5
#pragma GCC optimize(2) #pragma GCC optimize(3) #pragma GCC optimize("Ofast") #pragma GCC optimize("inline") #include <bits/stdc++.h> #define ll long long using namespace std; const int N=1e5+5; int n,m,blo,temp,nn,a[N],b[N], bl[N], cnt[N<<1], num[N<<1], l, r, mp[N<<1], tot, ans[N], cnt1, cnt2, tim; struct query {int l,r,tim,id; }q[N]; struct modify { int x, val, last; }c[N]; inline int read() { int x=0;char ch=getchar(); while(!isdigit(ch))ch=getchar(); while(isdigit(ch))x=x*10+(ch^48),ch=getchar(); return x; } inline int cmp(query a, query b) { return (bl[a.l] ^ bl[b.l]) ? bl[a.l] < bl[b.l] : ((bl[a.r] ^ bl[b.r]) ? bl[a.r] < bl[b.r] : a.tim < b.tim); } inline void add(int x) { --num[cnt[x]]; ++num[++cnt[x]]; } inline void del(int x) { --num[cnt[x]]; ++num[--cnt[x]]; } inline void change(int x, int y) { if (l<=x && x<=r) del(a[x]), add(y); a[x] = y; } inline void discrete() { sort(mp+1, mp+tot+1); nn=unique(mp+1, mp+tot+1)-mp; for (int i=1;i<=n;i++) a[i]=lower_bound(mp+1,mp+nn+1,a[i])-mp+1; for (int i=1;i<=cnt2;i++) c[i].last=lower_bound(mp+1,mp+nn+1,c[i].last)-mp+1, c[i].val=lower_bound(mp+1,mp+nn+1,c[i].val)-mp+1; } int main() { n=read(),m=read(); blo = pow(n, 2.0/3.0); for (int i=1;i<=n;i++) a[i]=read(), b[i] = mp[++tot] = a[i], bl[i]=(i-1)/blo; for (int i=1,op,l,r;i<=m;i++) { op=read(),l=read(),r=read(); if (op==1) q[++cnt1] = {l, r, tim, cnt1}; else ++tim, c[++cnt2] = {l, r, b[l]}, b[l] = r, mp[++tot] = r; } discrete(); sort(q+1, q+cnt1+1,cmp); r=0, l=1; num[0]=nn+1; tim=0; for (int i=1;i<=cnt1;i++) { while (r<q[i].r) add(a[++r]); while (l>q[i].l) add(a[--l]); while (r>q[i].r) del(a[r--]); while (l<q[i].l) del(a[l++]); while (tim<q[i].tim) tim++, change(c[tim].x, c[tim].val); while (tim>q[i].tim) change(c[tim].x, c[tim].last), --tim; int tmp=0; while (num[tmp]) ++tmp; ans[q[i].id] = tmp; } for (int i=1;i<=cnt1;i++) printf("%d\n", ans[i]); return 0; }
树上莫队
SP10707 COT2 - Count on a tree II
题目描述
给定一个n个节点的树,每个节点表示一个整数,问u到v的路径上有多少个不同的整数。
#include<cstdio> #include<cmath> #include<cctype> #include<cstring> #include<vector> #include<algorithm> using namespace std; const int N = 1e5 + 5; inline int read() { int x=0;char ch=getchar(); while(!isdigit(ch))ch=getchar(); while(isdigit(ch))x=x*10+(ch^48),ch=getchar(); return x; } int n, m,belong[N], block, a[N], b[N]; vector<int>v[N]; struct query {int l, r, id, lca, ans;}q[N]; inline int cmp(query a, query b) { return (belong[a.l] ^ belong[b.l]) ? belong[a.l] < belong[b.l] : ((belong[a.l] & 1) ? a.r < b.r : a.r > b.r); } inline void Discretization() { sort(b + 1, b + N + 1); int num = unique(b + 1, b + N + 1) - b - 1; for(int i = 1; i <= N; i++) a[i] = lower_bound(b + 1, b + num + 1, a[i]) - b; } int deep[N], top[N], fa[N], siz[N], son[N], st[N], ed[N], pot[N], tot; void dfs1(int x, int _fa) { fa[x] = _fa; siz[x] = 1; st[x] = ++ tot; pot[tot] = x; for(int i = 0; i < v[x].size(); i++) { int to = v[x][i]; if(deep[to]) continue; deep[to] = deep[x] + 1; dfs1(to, x); siz[x] += siz[to]; if(siz[to] > siz[son[x]]) son[x] = to; } ed[x] = ++tot; pot[tot] = x; } void dfs2(int x, int topfa) { top[x] = topfa; if(!son[x]) return ; dfs2(son[x], topfa); for(int i = 0; i < v[x].size(); i++) { int to = v[x][i]; if(top[to]) continue; dfs2(to, to); } } inline int Lca(int x, int y) { while(top[x] != top[y]) { if(deep[top[x]] < deep[top[y]]) swap(x, y); x = fa[top[x]]; } return deep[x] < deep[y] ? x : y; } int Ans, t[N], vis[N], happen[N]; inline void add(int x) { if (++happen[x] == 1) Ans++; } inline void del(int x) { if (--happen[x] == 0) Ans--; } inline void change(int x) { vis[x] ? del(a[x]) : add(a[x]); vis[x] ^= 1; } int main() { n = read(); m = read(); block = sqrt(n); for (int i = 1; i <= n; i++) a[i] = b[i] = read(); for (int i = 1; i <= n * 2; i++) belong[i] = i / block + 1; Discretization(); for (int i = 1; i < n; i++) { int x = read(), y = read(); v[x].push_back(y); v[y].push_back(x); } deep[1] = 1; dfs1(1, 0); dfs2(1, 1); for (int i = 1; i <= m; i++) { int x = read(), y = read(); if (st[x] > st[y]) swap(x, y); int _lca =Lca(x, y); q[i].id = i; if (_lca == x) q[i].l = st[x], q[i]. r = st[y]; else q[i].l = ed[x], q[i].r = st[y], q[i].lca = _lca; } sort(q + 1, q + m + 1,cmp); int l = 1, r = 0; for(int i = 1; i <= m; i++) { while(l < q[i].l) change(pot[l]), l++; while(l > q[i].l) l--, change(pot[l]); while(r < q[i].r) r++, change(pot[r]); while(r > q[i].r) change(pot[r]), r--; if (q[i].lca) change(q[i].lca); q[i].ans = Ans; if (q[i].lca) change(q[i].lca); } for(int i = 1; i <= m; i++) t[q[i].id] = q[i].ans; for(int i = 1; i <= m; i++) printf("%d\n", t[i]); return 0; }
树上带修莫队
P4074 [WC2013]糖果公园
#pragma GCC optimize(2) #pragma GCC optimize(3) #pragma GCC optimize("Ofast") #pragma GCC optimize("inline") #include<bits/stdc++.h> #define rg register using namespace std; typedef long long ll; const int N=1e5+5; int n,m,Q,cnte,cnt1,cnt2,tim; int head[N],val[N],w[N],a[N],b[N],dep[N]; int f[N][20]; int bl[N<<1],cnt[N]; ll nowans,ans[N]; bool vis[N]; inline int read() { int x=0;char ch=getchar(); while(!isdigit(ch))ch=getchar(); while(isdigit(ch))x=x*10+(ch^48),ch=getchar(); return x; } struct edge{int to,nxt;}e[N<<1]; inline void add(int u,int v) { e[++cnte].nxt=head[u]; head[u]=cnte; e[cnte].to=v; } int ouler[N<<1],st[N],ed[N]; inline void dfs(int x,int fa) { for (rg int i=1;i<=18;i++) f[x][i]=f[f[x][i-1]][i-1]; dep[x]=dep[fa]+1; ouler[++tim]=x;st[x]=tim; for (rg int i=head[x];i;i=e[i].nxt) { int y=e[i].to; if(y==fa)continue; f[y][0]=x;dfs(y,x); } ouler[++tim]=x;ed[x]=tim; } inline int lca(int x,int y) { if (dep[x]>dep[y]) swap(x,y); for (int i=18;i>=0;i--) if (dep[f[y][i]]>=dep[x]) y=f[y][i]; if (x==y) return x; for (rg int i=18;i>=0;i--) if (f[x][i]!=f[y][i]) x=f[x][i],y=f[y][i]; return f[x][0]; } struct query{int l,r,lca,id,tim;}q[N]; struct modify{int pos,last,now;}c[N]; inline int cmp(query a, query b) { return (bl[a.l] ^ bl[b.l]) ? bl[a.l] < bl[b.l] : ((bl[a.l] & 1) ? a.r < b.r : a.r > b.r); } inline void del(int c) {nowans-=1ll*val[c]*w[cnt[c]--];} inline void add(int c) {nowans+=1ll*val[c]*w[++cnt[c]];} inline void change(int pos,int k) { if (vis[pos]) del(a[pos]),add(k); a[pos]=k; } inline void work(int pos) { if (vis[ouler[pos]]) del(a[ouler[pos]]); else add(a[ouler[pos]]); vis[ouler[pos]]^=1; } int main() { n=read(),m=read(),Q=read(); for (rg int i=1;i<=m;i++) val[i]=read(); for (rg int i=1;i<=n;i++) w[i]=read(); for (rg int i=1;i<n;i++) { int u=read(),v=read(); add(u,v),add(v,u); } dfs(1,0); for (rg int i=1;i<=n;i++) a[i]=b[i]=read(); for (rg int i=1;i<=Q;i++) { int op=read(),x=read(),y=read(); if (op) q[++cnt1]=(query){x,y,0,cnt1,cnt2}; else c[++cnt2]=(modify){x,b[x],y},b[x]=y; } for (rg int i=1;i<=Q;i++) { if (st[q[i].l]>st[q[i].r]) swap(q[i].l,q[i].r); int z=lca(q[i].l,q[i].r); if (z==q[i].l) q[i].l=st[q[i].l],q[i].r=st[q[i].r]; else q[i].l=ed[q[i].l],q[i].r=st[q[i].r],q[i].lca=z; } int blo=pow(2*n,2.0/3.0); for (rg int i=1;i<=2*n;i++) bl[i]=(i-1)/blo+1; sort(q+1,q+cnt1+1,cmp); int l=1,r=0,tim1=0; for (rg int i=1;i<=cnt1;i++) { while (tim1<q[i].tim) change(c[tim1+1].pos,c[tim1+1].now),tim1++; while (tim1>q[i].tim) change(c[tim1].pos,c[tim1].last),tim1--; while (l<q[i].l) work(l++); while (l>q[i].l) work(--l); while (r<q[i].r) work(++r); while (r>q[i].r) work(r--); if (q[i].lca) work(st[q[i].lca]); ans[q[i].id]=nowans; if (q[i].lca) work(st[q[i].lca]); } for (rg int i=1;i<=cnt1;i++) printf("%lld\n",ans[i]); return 0; }
回滚莫队
AT1219 歴史の研究
给定一个长度为n的序列,离线询问m个问题,每次回答区间内元素权值乘以元素出现次数的最大值。
#include <bits/stdc++.h> using namespace std; const int N = 1e5+5; int n,m,sz,T,raw[N],val[N],t,cnt[N],cnt_[N]; int bl[N],L[N],R[N]; long long ans[N],Max,a[N]; struct query{int l,r,id;}q[N]; inline int cmp(query a, query b) { return (bl[a.l] ^ bl[b.l]) ? bl[a.l] < bl[b.l] : a.r < b.r; } inline void discrete() { sort( raw+1 , raw+t+1 ); t = unique( raw+1 , raw+t+1 ) - (raw+1); for (int i=1;i<=n;i++) val[i] = lower_bound( raw+1 , raw+t+1 , a[i] ) - raw; } inline void add(int p,long long &maxval) { cnt[val[p]]++; maxval = max( maxval , 1ll * cnt[val[p]] * a[p] ); } inline void del(int p) {cnt[val[p]]--;} int main() { scanf("%d%d",&n,&m); for (int i=1;i<=n;i++) scanf("%lld",&a[i]) , raw[++t] = a[i]; for (int i=1;i<=m;i++) scanf("%d%d",&q[i].l,&q[i].r) , q[i].id = i; discrete(); sz = sqrt(n) , T = n/sz; for (int i=1;i<=T;i++) { if ( i * sz > n ) break; L[i] = (i-1) * sz + 1; R[i] = i * sz; } if ( R[T] < n ) T++ , L[T] = R[T-1] + 1 , R[T] = n; for (int i=1;i<=T;i++) for (int j=L[i];j<=R[i];j++) bl[j] = i; sort( q+1 , q+m+1 , cmp ); int l = 1 , r = 0 , last = 0; for (int i=1;i<=m;i++) { if ( bl[q[i].l] == bl[q[i].r] ) { for (int j=q[i].l;j<=q[i].r;j++) cnt_[val[j]]++; long long temp = 0; for (int j=q[i].l;j<=q[i].r;j++) temp = max( temp , 1ll * cnt_[val[j]] * a[j] ); for (int j=q[i].l;j<=q[i].r;j++) cnt_[val[j]]--; ans[ q[i].id ] = temp; continue; } if ( last ^ bl[q[i].l] ) { while ( r > R[bl[q[i].l]] ) del(r--); while ( l < R[bl[q[i].l]]+1 ) del(l++); Max = 0 , last = bl[q[i].l]; } while ( r < q[i].r ) add(++r,Max); long long temp = Max; int l_ = l; while ( l_ > q[i].l ) add(--l_,temp); while ( l_ < l ) del(l_++); ans[ q[i].id ] = temp; } for (int i=1;i<=m;i++) printf("%lld\n",ans[i]); return 0; }
愿你走出半生,归来仍是少年