「some」分块
乱写
分块
分块应该算是一种思想,而不是数据结构。
基本思想: 通过对原数据进行划分,也就是所谓的分块,并通过预处理块内信息,从而获得较暴力更优的时间复杂度。 还是挺暴力的
时间复杂度: 分块的时间复杂度取决于块长,调整块长可以使修改查询时散块与整块的数量发生改变。可以通过均值不等式来计算出在某个问题下的最优块长,但在一般情况下,块长都设为 \(\sqrt{n}\) 。
对于一些满足区间加减性的信息,显然可以用线段树或树状数组解决,分块同样可以解决,但分块并不局限于此,其更加灵活。但是复杂度堪忧
部分摘自oi-wiki
区间修改
维护类似于线段树中懒标记的东西,更新时散块暴力修改,整块打标记,查询时把标记释放即可。
区间加减
最简单的一类,直接维护个加法标记,修改更新,查询释放,没啥好说的。
区间加+单点查询
区间加+区间查询
单点加+区间查询
只是因为没地方放所以塞到这里了
区间乘
之前那个线段树的区间乘就没写出来,先咕了。
upd on 09-12
题目中一共有两个操作,区间乘,区间加,所以维护两个标记,一个 \(mul\) 乘法标记,一个 \(add\) 加法标记,维护难点在于标记优先级问题,无非就是两种:
- 先加后乘,即 \((val+add)\times mul\) 。
- 先乘后加,即 \(val\times mul+add\) 。
如果是先加后乘,给 \(val\) 加上一个 \(k\) 的话,此时按1直接来算就错了,想要正确还要修改 \(mul\) 的值,过于麻烦。其实是不太会修改
如果是先乘后加,在给 \(val\) 加上一个 \(k\) 时直接修改 \(add\) 即可,乘上一个 \(k\) 只需同时给 \(mul,add\) 同时乘 \(k\) 即可保证正确性。
所以选择先乘后加。
那这题就没了。调试极度ex
这个是求单点值,求区间和的话再维护个块内和就行了。
更难调了
区间开方
还没去写....其实是不会
upd on 09-11 爬回来补上了
直接ODT就可以草过去。
ODT's main code
inline void modify(int l,int r,int val = 0)
{
iter itr = split(r+1),itl = split(l);
for(iter it=itl; it!=itr; it++)
{
it->val = sqrt(it->val);
if(val==it->val)
{ r = it->r; }
else
{
if(val){ assign(l,r,val); }
l = it->l,r = it->r,val = it->val;
}
}
assign(l,r,val);
}
发现有作用的开方操作其实很少,开几次就会变成1,所以可以开个 \(vis\) 数组记录一下当前这个块是否全为1,修改时判断是否已经全为1,是就不用修改了,否则直接去枚举区间内所有元素,因为实际更新操作很少,所以并不会TLE, \(vis\) 数组跟着更新一下即可。
求区间和则是常规操作,不再赘述。
Code
#include<cmath>
#include<cstdio>
#define MAX 50003
#define int long long
const int LEN = 230;
#define re register
namespace some
{
struct stream
{
template<typename type>inline stream &operator >>(type &s)
{
int w=1; s=0; char ch=getchar();
while(ch<'0'||ch>'9'){ if(ch=='-')w=-1; ch=getchar(); }
while(ch>='0'&&ch<='9'){ s=s*10+ch-'0'; ch=getchar(); }
return s*=w,*this;
}
}cin;
auto min = [](int a,int b) -> int { return a<b?a:b; };
}using namespace some;
namespace OMA
{
int n,a[MAX];
bool vis[LEN];
int len,id[MAX];
int sum[LEN],L[LEN],R[LEN];
auto update = [](int l,int r,int p) -> void
{
if(vis[p])
{ return ; }
vis[p] = true;
//printf("begin\n");
for(re int i=l; i<=r; i++)
{ sum[p] -= a[i],/*printf("1: %lld ",a[i]),*/a[i] = sqrt(a[i]),/*printf("2: %lld\n",a[i]),*/sum[p] += a[i]; }
for(re int i=L[p]; i<=R[p]; i++)
{ if(a[i]>1){ vis[p] = false; break; } }
//printf("sum=%lld\nend\n",sum[p]);
};
auto modify = [](int l,int r) -> void
{
int p1 = id[l],p2 = id[r];
if(p1==p2)
{ update(l,r,p1); return ; }
update(l,R[p1],p1);
for(re int i=p1+1; i<=p2-1; i++)
{ update(L[i],R[i],i); }
update(L[p2],r,p2);
};
auto query = [](int l,int r) -> int
{
//printf("query\n");
int res = 0;
int p1 = id[l],p2 = id[r];
//printf("%lld %lld\n",p1,p2);
if(p1==p2)
{
for(re int i=l; i<=r; i++)
{ res += a[i]; /*printf("%lld ",a[i]);*/ }
//printf("\n");
return res;
}
for(re int i=l; id[i]==p1; i++)
{ res += a[i]; }
for(re int i=p1+1; i<=p2-1; i++)
{ res += sum[i]; }
for(re int i=r; id[i]==p2; i--)
{ res += a[i]; }
return res;
};
int m;
auto main = []() -> signed
{
//freopen("my.out","w",stdout);
cin >> n; len = sqrt(n);
for(re int i=1; i<=n; i++)
{ cin >> a[i]; sum[id[i] = (i-1)/len+1] += a[i]; }
for(re int i=1; i<=id[n]; i++)
{ L[i] = R[i-1]+1,R[i] = min(i*len,n); }
for(re int i=1,opt,l,r,c; i<=n; i++)
{
cin >> opt >> l >> r >> c;
if(l>r)
{ int t=l; l=r; r=t; }
if(opt==1)
{ printf("%lld\n",query(l,r)); }
else
{ modify(l,r); }
}
return 0;
};
}
signed main()
{ return OMA::main(); }
loj的花神 数据里有权值全为0的,判1的时候一块判上即可。
区间赋值
直接ODT(珂朵莉树)就可以草过去
ODT's main code
inline int query(int l,int r,int c)
{
int res = 0;
iter itr = split(r+1),itl = split(l);
for(auto it=itl; it!=itr; it++)
{ if(it->val==c){ res += it->r-it->l+1; } }
return res;
}
维护一个 \(tag\) 数组,表示当前块是否被整体赋值了,再维护一个 \(val\) 数组,表示被修改成了那个值,散块查询时下放标记,整块判断 \(val\) 数组是否等于修改的值即可,是直接更新答案,不是的话块内二分解决。
Code
#include<cmath>
#include<cstdio>
#include<cctype>
#include<algorithm>
#define MAX 100003
const int LEN = 320;
#define re register
using std::sort;
using std::lower_bound;
using std::upper_bound;
namespace some
{
struct stream
{
template<typename type>inline stream &operator >>(type &s)
{
bool w=0; s=0; char ch=getchar();
while(!isdigit(ch)){ w|=ch=='-'; ch=getchar(); }
while(isdigit(ch)){ s=(s<<1)+(s<<3)+(ch^48); ch=getchar(); }
return s=w?-s:s,*this;
}
}cin;
auto min = [](int a,int b) -> int { return a<b?a:b; };
}using namespace some;
namespace OMA
{
int n,a[MAX];
bool tag[LEN];
int val[LEN];
int len,id[MAX];
int L[LEN],R[LEN],ord[LEN][LEN];
auto Push_down = [](int p) -> void
{
if(!tag[p])
{ return ; }
for(re int i=L[p]; i<=R[p]; i++)
{ a[i] = val[p]; }
tag[p] = false;
};
auto update = [](int p) -> void
{
for(re int i=L[p]; i<=R[p]; i++)
{ ord[p][i-L[p]+1] = a[i]; }
sort(ord[p]+1,ord[p]+R[p]-L[p]+1+1);
};
auto doit = [](int l,int r,int c,int res = 0) -> int
{
int p1 = id[l],p2 = id[r];
if(p1==p2)
{
Push_down(p1);
for(re int i=l; i<=r; i++)
{ res += a[i]==c,a[i] = c; }
//update(p1);
return res;
}
Push_down(p1);
for(re int i=l; id[i]==p1; i++)
{ res += a[i]==c,a[i] = c; }
//update(p1);
for(re int i=p1+1,rec1,rec2; i<=p2-1; i++)
{
if(tag[i]&&val[i]==c)
{ res += len; }
else if(!tag[i])
{
update(i);
rec1 = upper_bound(ord[i]+1,ord[i]+1+len,c)-ord[i];
rec2 = lower_bound(ord[i]+1,ord[i]+1+len,c)-ord[i];
res += rec1-rec2;
}
tag[i] = true,val[i] = c;
}
Push_down(p2);
for(re int i=r; id[i]==p2; i--)
{ res += a[i]==c,a[i] = c; }
//update(p2);
return res;
};
auto main = []() -> signed
{
cin >> n; len = sqrt(n);
for(re int i=1; i<=n; i++)
{ cin >> a[i]; id[i] = (i-1)/len+1; }
for(re int i=1; i<=id[n]; i++)
{ L[i] = R[i-1]+1,R[i] = min(i*len,n); update(i); }
for(re int i=1,l,r,c; i<=n; i++)
{ cin >> l >> r >> c; printf("%d\n",doit(l,r,c)); /*write(doit(l,r,c));*/ }
return 0;
};
}
signed main()
{ return OMA::main(); }
区间众数
离线的话直接用莫队就好了,然而有强制在线的......
典型在线区间众数问题。
如果直接写暴力的话,便是开个桶,统计个数统计答案,考虑如何更高效不是特别暴力的来处理这个问题。
分块。
首先,对于给定一个区间的众数,其来源只有两个:
-
区间中所有整块中的众数
-
出现在两个散块之间的数
那么就有了分块所要维护的信息
对于散块: 沿用暴力思路,直接统计。
对于整块: 可以预处理出第 \(i\) 个块到 第\(j\) 个块之间的众数。
具体来说:
先将权值离散化,设 \(cnt_{i,j}\) 表示前 \(i\) 个块中, \(j\) 出现的次数, 设\(mode_{i,j}\) 表示第 \(i\) 个块 到 第 \(j\) 个块之间的众数。
这样的话,可以通过 \(O(\frac{n^{2}}{T})\) 来预处理出 \(mode\) 数组,查询时,整块直接通过 \(mode\) 数组得到,\(O(1)\) ,散块暴力枚举统计,\(O(T)\),所以总查询 \(O(mT)\)
设 \(m\) 与 \(n\) 同阶,那么总复杂度 \(O(\frac{n^{2}}{T}+nT)\) ,由均值不等式 \(\frac{n^{2}}{T}=nT\) ,当 \(T=\sqrt{n}\) 时,复杂度最优。
一些细节见code。
Code
#include<cmath>
#include<cstdio>
#include<algorithm>
#define MAX 40010
const int LEN = 210;
#define re register
using std::sort;
using std::unique;
using std::lower_bound;
namespace some
{
struct stream
{
template<typename type>inline stream &operator >>(type &s)
{
int w=1; s=0; char ch=getchar();
while(ch<'0'||ch>'9'){ if(ch=='-')w=-1; ch=getchar(); }
while(ch>='0'&&ch<='9'){ s=s*10+ch-'0'; ch=getchar(); }
return s*=w,*this;
}
}cin;
inline int min(int a,int b)
{ return a<b?a:b; }
inline void swap(int &a,int &b)
{ int t=a; a=b; b=t; }
}using namespace some;
namespace OMA
{
int len,id[MAX];
int n,m,x,a[MAX];
int val[MAX],tot;
int cnt[LEN][MAX],tub[MAX];
int L[LEN],R[LEN],mode[LEN][LEN];
struct BLOCK
{
inline int query(int l,int r)
{
int p1 = id[l],p2 = id[r];
int rec,res = mode[p1+1][p2-1];
if(p1==p2)
{
for(re int i=l; i<=r; i++)
{ tub[a[i]]++; }
for(re int i=l; i<=r; i++)
{
if(tub[a[i]]>tub[res])
{ res = a[i]; }
if(a[i]<res&&tub[a[i]]==tub[res])
{ res = a[i]; }
}
for(re int i=l; i<=r; i++)
{ tub[a[i]] = 0; }
return res;
}
tub[rec = res] = cnt[p2-1][res]-cnt[p1][res];
for(re int i=l; id[i]==p1; i++)
{
if(!tub[a[i]])
{ tub[a[i]] = cnt[p2-1][a[i]]-cnt[p1][a[i]]; }
tub[a[i]]++;
}
for(re int i=r; id[i]==p2; i--)
{
if(!tub[a[i]])
{ tub[a[i]] = cnt[p2-1][a[i]]-cnt[p1][a[i]]; }
tub[a[i]]++;
}
for(re int i=l; id[i]==p1; i++)
{
if(tub[a[i]]>tub[res])
{ res = a[i]; }
else if(a[i]<res&&tub[a[i]]==tub[res])
{ res = a[i]; }
}
for(re int i=r; id[i]==p2; i--)
{
if(tub[a[i]]>tub[res])
{ res = a[i]; }
else if(a[i]<res&&tub[a[i]]==tub[res])
{ res = a[i]; }
}
tub[rec] = 0;
for(re int i=l; id[i]==p1; i++)
{ tub[a[i]] = 0; }
for(re int i=r; id[i]==p2; i--)
{ tub[a[i]] = 0; }
return res;
}
}block;
signed main()
{
//freopen("node.in","r",stdin);
//freopen("my.out","w",stdout);
cin >> n >> m; len = sqrt(n);
for(re int i=1; i<=n; i++)
{ cin >> a[i]; val[i] = a[i]; id[i] = (i-1)/len+1; }
sort(val+1,val+1+n);
tot = unique(val+1,val+1+n)-val-1;
for(re int i=1; i<=n; i++)
{ a[i] = lower_bound(val+1,val+1+tot,a[i])-val; }
for(re int i=1; i<=id[n]; i++)
{
L[i] = R[i-1]+1,R[i] = min(i*len,n);
for(re int j=L[i]; j<=R[i]; j++)
{ cnt[i][a[j]]++; }
for(re int j=1; j<=tot; j++)
{ cnt[i][j] += cnt[i-1][j]; }
}
for(re int i=1; i<=id[n]; i++)
{
for(re int j=i; j<=id[n]; j++)
{
int xam = mode[i][j-1];
for(re int k=L[j]; k<=R[j]; k++)
{
if(cnt[j][a[k]]-cnt[i-1][a[k]]>cnt[j][xam]-cnt[i-1][xam])
{ xam = a[k]; }
if(a[k]<xam&&cnt[j][a[k]]-cnt[i-1][a[k]]==cnt[j][xam]-cnt[i-1][xam])
{ xam = a[k]; }
}
mode[i][j] = xam;
}
}
for(re int i=1,l,r; i<=m; i++)
{
cin >> l >> r;
l = (l+x-1)%n+1,r = (r+x-1)%n+1;
if(l>r)
{ swap(l,r); }
printf("%d\n",x = val[block.query(l,r)]);
}
return 0;
}
}
signed main()
{ return OMA::main(); }
离线版:数列分块9
没有强制在线,可以用回滚莫队来写,但现在是分块,所以就拿分块来做。
跟上一道题差不多,用跟蒲公英一样的套路即可。
一样的离散化,设 \(cnt_{i,j}\) 表示前i个块中,事件j的出现次数,\(xam_{i,j}\) 表示第 \(i\) 个块到第 \(j\) 个块的答案。两个数组可以通过预处理得到。
查询时散块暴力,整块通过 \(xam\) 数组得到答案。
但这回查询时没那么ex,直接统计出现次数,一乘一比即可。
然而还是调了半天的sb错误,指预处理时数组下标打错,made
Code
#include<cmath>
#include<cstdio>
#include<algorithm>
#define MAX 100010
#define int64_t long long
const int LEN = 320;
#define re register
using std::sort;
using std::unique;
using std::lower_bound;
namespace some
{
struct stream
{
template<typename type>inline stream &operator >>(type &s)
{
int w=1; s=0; char ch=getchar();
while(ch<'0'||ch>'9'){ if(ch=='-')w=-1; ch=getchar(); }
while(ch>='0'&&ch<='9'){ s=s*10+ch-'0'; ch=getchar(); }
return s*=w,*this;
}
}cin;
template<typename type>inline type max(type a,type b)
{ return a>b?a:b; }
template<typename type>inline type min(type a,type b)
{ return a<b?a:b; }
}using namespace some;
namespace OMA
{
int n,q,x[MAX];
int len,id[MAX];
int tot,val[MAX];
int L[LEN],R[LEN];
int64_t xam[LEN][LEN];
int cnt[LEN][MAX],tub[MAX];
struct BLOCK
{
inline int64_t query(int l,int r)
{
int p1 = id[l],p2 = id[r];
int64_t res = xam[p1+1][p2-1];
if(p1==p2)
{
for(re int i=l; i<=r; i++)
{ tub[x[i]]++; }
for(re int i=l; i<=r; i++)
{
res = max(res,1LL*tub[x[i]]*val[x[i]]);
tub[x[i]] = 0;
}
return res;
}
//printf("%lld\n",res);
for(re int i=l; id[i]==p1; i++)
{
if(!tub[x[i]])
{ tub[x[i]] = cnt[p2-1][x[i]]-cnt[p1][x[i]]; }
tub[x[i]]++;
}
for(re int i=r; id[i]==p2; i--)
{
if(!tub[x[i]])
{ tub[x[i]] = cnt[p2-1][x[i]]-cnt[p1][x[i]]; }
tub[x[i]]++;
}
for(re int i=l; id[i]==p1; i++)
{ res = max(res,1LL*tub[x[i]]*val[x[i]]); }
//printf("res1=%lld ",res);
for(re int i=r; id[i]==p2; i--)
{ res = max(res,1LL*tub[x[i]]*val[x[i]]); }
//printf("res2=%lld\n",res);
for(re int i=l; id[i]==p1; i++)
{ tub[x[i]] = 0; }
for(re int i=r; id[i]==p2; i--)
{ tub[x[i]] = 0; }
return res;
}
}block;
signed main()
{
//freopen("node.in","r",stdin);
//freopen("my.out","w",stdout);
cin >> n >> q; len = sqrt(n);
for(re int i=1; i<=n; i++)
{ cin >> x[i]; val[i] = x[i]; id[i] = (i-1)/len+1; }
sort(val+1,val+1+n);
tot = unique(val+1,val+1+n)-val-1;
for(re int i=1; i<=n; i++)
{ x[i] = lower_bound(val+1,val+1+tot,x[i])-val; }
for(re int i=1; i<=id[n]; i++)
{
L[i] = R[i-1]+1,R[i] = min(i*len,n);
for(re int j=L[i]; j<=R[i]; j++)
{ cnt[i][x[j]]++; }
for(re int j=1; j<=tot; j++)
{ cnt[i][j] += cnt[i-1][j]; }
}
for(re int i=1; i<=id[n]; i++)
{
for(re int j=i; j<=id[n]; j++)
{
int64_t res = xam[i][j-1];
for(re int k=L[j]; k<=R[j]; k++)
{ res = max(res,1LL*(cnt[j][x[k]]-cnt[i-1][x[k]])*val[x[k]]); }
xam[i][j] = res;
}
}
for(re int i=1,l,r; i<=q; i++)
{
cin >> l >> r;
printf("%lld\n",block.query(l,r));
}
return 0;
}
}
signed main()
{ return OMA::main(); }
还有个带修改的区间众数,然而并不会,具体可以参考陈立杰的区间众数解题报告。
块内二分
主要是用于解决一类查询满足大小关系的数(或者个数)。
对于每个块,维护块内的有序关系,即开个数组记录每个块中的元素,每回修改时 \(sort\) 一遍,修改时散块更新,整块打标记,查询时散块暴力查找,整块可以直接通过二分来解决。
区间加+查询小于x的个数
区间加+查询大于等于x的个数
然而70pts没调出来
调出来了,made,整块二分查找的时候忘记把标记加上去了,淦。
区间加+查询前驱
Code
#include<cmath>
#include<cstdio>
#include<climits>
#include<algorithm>
#define MAX 100010
#define INF INT_MIN
const int LEN = 320;
#define re register
using std::sort;
namespace some
{
struct stream
{
template<typename type>inline stream &operator >>(type &s)
{
int w=1; s=0; char ch=getchar();
while(ch<'0'||ch>'9'){ if(ch=='-')w=-1; ch=getchar(); }
while(ch>='0'&&ch<='9'){ s=s*10+ch-'0'; ch=getchar(); }
return s*=w,*this;
}
}cin;
inline int max(int a,int b)
{ return a>b?a:b; }
inline int min(int a,int b)
{ return a<b?a:b; }
}using namespace some;
namespace OMA
{
int n,a[MAX];
int len,id[MAX];
int L[LEN],R[LEN];
struct BLOCK
{
int delta[LEN];
int ord[LEN][LEN];
inline void update(int p)
{
for(re int i=L[p]; i<=R[p]; i++)
{ ord[p][i-L[p]+1] = a[i]; }
sort(ord[p]+1,ord[p]+R[p]-L[p]+1+1);
/*for(re int i=1; i<=R[p]-L[p]+1; i++)
{ printf("%d ",ord[p][i]); }
printf("\n");*/
}
inline void modify(int l,int r,int c)
{
int p1 = id[l],p2 = id[r];
if(p1==p2)
{
for(re int i=l; i<=r; i++)
{ a[i] += c; }
update(p1);
return ;
}
for(re int i=l; id[i]==p1; i++)
{ a[i] += c; }
for(re int i=p1+1; i<=p2-1; i++)
{ delta[i] += c; }
for(re int i=r; id[i]==p2; i--)
{ a[i] += c; }
update(p1),update(p2);
}
inline int binary_search(int p,int c)
{
int rec = INF,l = 1,r = len;
while(l<=r)
{
int mid = (l+r)>>1;
if(ord[p][mid]<c)
{ l = mid+1,rec = max(rec,ord[p][mid]); }
else
{ r = mid-1; }
}
return rec!=INF?rec+delta[p]:rec;
}
inline int query(int l,int r,int c)
{
int res = INF,p1 = id[l],p2= id[r];
if(p1==p2)
{
for(re int i=l; i<=r; i++)
{ if(a[i]+delta[p1]<c){ res = max(res,a[i]+delta[p1]); } }
return res==INF?-1:res;
}
for(re int i=l; id[i]==p1; i++)
{ if(a[i]+delta[p1]<c){ res = max(res,a[i]+delta[p1]); } }
for(re int i=p1+1; i<=p2-1; i++)
{ res = max(res,binary_search(i,c-delta[i])); }
for(re int i=r; id[i]==p2; i--)
{ if(a[i]+delta[p2]<c){ res = max(res,a[i]+delta[p2]); } }
return res==INF?-1:res;
}
}block;
signed main()
{
cin >> n; len = sqrt(n);
for(re int i=1; i<=n; i++)
{ cin >> a[i]; id[i] = (i-1)/len+1; }
for(re int i=1; i<=id[n]; i++)
{
L[i] = R[i-1]+1,R[i] = min(i*len,n);
block.update(i);
}
for(re int i=1,opt,l,r,c; i<=n; i++)
{
cin >> opt >> l >> r >> c;
if(!opt)
{ block.modify(l,r,c); }
else
{ printf("%d\n",block.query(l,r,c)); }
}
return 0;
}
}
signed main()
{ return OMA::main(); }
用分块的话是比较有意思的一道题,用莫队就是板子题
按照分块一般的套路,预处理出整块的答案,散块暴力搞,本题中一个很直接的处理方法是预处理出前 \(i\) 个块中 颜色 \(j\) 出现的次数,再处理出每个整块的答案,查询直接开个桶暴力统计散块,写的时候发现处理整块并不好处理 我不会,考虑换种处理方法来做。
块内二分。
如何用块内二分来做?
一个很妙的想法,我们用 \(pre_{i}\) 表示 \(i\) 这个位置上的颜色上次出现的位置。
这样的话对于一个查询的区间 \([l,r]\) ,如果当前位置 \(i\) 的 \(pre_{i}\) 小于 \(l\) 则说明其在当前查询区间内为第一次出现,这样就可以用块内二分来做,对于每个整块,维护块内 \(pre\) 的有序,这样的话整块查询只需二分即可,散块暴力统计。
修改的时候,直接再处理一边即可,修改时有个 \(sort\) 所以单次修改复杂度 \(O(n+n\log{\sqrt{n}})\),查询时有个二分,所以单次查询 \(O(\sqrt{n}\log{\sqrt{n}})\) 。
Code
#include<cmath>
#include<cstdio>
#include<algorithm>
#define MAX 133333
const int LEN = 371;
#define re register
using std::sort;
using std::lower_bound;
namespace some
{
struct stream
{
template<typename type>inline stream &operator >>(type &s)
{
int w=1; s=0; char ch=getchar();
while(ch<'0'||ch>'9'){ if(ch=='-')w=-1; ch=getchar(); }
while(ch>='0'&&ch<='9'){ s=s*10+ch-'0'; ch=getchar(); }
return s*=w,*this;
}
}cin;
inline int min(int a,int b)
{ return a<b?a:b; }
}using namespace some;
namespace OMA
{
int n,m,c[MAX];
int id[MAX],len;
int L[LEN],R[LEN];
int pre[MAX],last[MAX*10];
struct BLOCK
{
int ord[LEN][LEN];
inline void modify()
{
for(re int i=1; i<=n; i++)
{ last[c[i]] = 0; }
for(re int i=1; i<=id[n]; i++)
{
for(re int j=L[i]; j<=R[i]; j++)
{ ord[i][j-L[i]+1] = pre[j] = last[c[j]],last[c[j]] = j; }
sort(ord[i]+1,ord[i]+R[i]-L[i]+1+1);
}
}
inline int query(int l,int r)
{
int res = 0,p1 = id[l],p2 = id[r];
if(p1==p2)
{
for(re int i=l; i<=r; i++)
{ if(pre[i]<l){ res++; } }
return res;
}
for(re int i=l; id[i]==p1; i++)
{ if(pre[i]<l){ res++; } }
for(re int i=p1+1; i<=p2-1; i++)
{ res += lower_bound(ord[i]+1,ord[i]+R[i]-L[i]+1+1,l)-ord[i]-1; }
for(re int i=r; id[i]==p2; i--)
{ if(pre[i]<l){ res++; } }
return res;
}
}block;
signed main()
{
cin >> n >> m; len = sqrt(n);
for(re int i=1; i<=n; i++)
{ cin >> c[i]; id[i] = (i-1)/len+1; }
for(re int i=1; i<=id[n]; i++)
{ L[i] = R[i-1]+1,R[i] = min(i*len,n); }
block.modify();
for(re int i=1,l,r; i<=m; i++)
{
char opt[5];
scanf("%s",opt);
cin >> l >> r;
if(opt[0]=='R')
{ c[l] = r,block.modify(); }
else
{ printf("%d\n",block.query(l,r)); }
}
return 0;
}
}
signed main()
{ return OMA::main(); }
加这道题的原因是写主席树时碰到一道不带修改的强制在线的统计颜色题,我写了个莫队预处理整块,查询用分块的sb方法60pts暴力50pts,预处理复杂度 \(O(n^2)\) 过高死了,所以决定学一下正经的分块写法。
不过那题 \(n,m,col\le 500000\) 的数据竟然过了,主要原因是没有修改QAQ。
有修改并且数据还大的话,离线带修莫队,在线我不会(逃)。貌似可以用树套树,然而不太会
块状链表
待填........
upd on 09-13
发现这玩意好像没啥用,直接用平衡树就可以,有 \(\log\) 我还要什么 \(\sqrt{}\) 呢?
然而POJ的那道不知为何一直re.....
值域分块
值域分块,顾名思义,就是按值域分块,与之前分块的区别就是由分序列变成了分值域,散块枚举由位置变成了权值,支持 \(O(1)\) 修改,\(O(T)\) 查询,其中 \(T\) 为块的大小。
莫队+值域分块
一共有两问,一个是个数,一个是种类数,个数对于分块轻轻松松,种类数对于莫队轻轻松松,所以考虑用这俩来解决该问题。
然而丧心病狂的是,它在区间限制的基础上,还有个值域的限制,这样的话就导致上边的做法就显得不太可行。
一个比较naive的想法是块内二分来维护有值域限制的个数,种类数可以再开一维来记录,但这样的话时间复杂度不太可以接受虽然时限10s。
莫队本质上是做了 \(n\sqrt{m}\) 次修改,\(m\) 次查询,所以修改必须是\(O(1)\),而查询次数相对较少,可以是 \(O(\sqrt{n})\) 的,所以用值域分块来解决这个问题。
先用莫队统计出当前区间\([l,r]\)内值的个数和种类,具体来说,可以开三个桶 \(cnt1_{i}\) 表示当前权值 \(i\) 的出现次数,\(cnt2_{i}\) 表示在 \(i\) 这个块出现了多少个数,\(cnt3_{i}\) 表示在 \(i\) 这个块出现了多少种数,注意,这里的块都是按值域划分的。然后再用分块去查询满足在 \([a,b]\) 中的答案即可
复杂度 \(O(n\sqrt{m}+m\sqrt{n})\) 。
由于数据过于水,序列里的数都比n小,所以离散化都不用。
Code
#include<cmath>
#include<cstdio>
#include<algorithm>
#define MAX 100010
const int LEN = 320;
#define re register
using std::sort;
namespace some
{
struct stream
{
template<typename type>inline stream &operator >>(type &s)
{
int w=1; s=0; char ch=getchar();
while(ch<'0'||ch>'9'){ if(ch=='-')w=-1; ch=getchar(); }
while(ch>='0'&&ch<='9'){ s=s*10+ch-'0'; ch=getchar(); }
return s*=w,*this;
}
}cin;
inline int min(int a,int b)
{ return a<b?a:b; }
}using namespace some;
namespace OMA
{
int n,m,val[MAX];
int ans1[MAX*10],ans2[MAX*10];
int cnt1[MAX],cnt2[LEN],cnt3[LEN];
int len,id[MAX]/*,L[LEN],R[LEN]*/;
struct mo_team
{
int l,r,a,b,p;
inline friend bool operator <(const mo_team &a,const mo_team &b)
{ return id[a.l]==id[b.l]?(id[a.l]&1?id[a.r]<id[b.r]:id[a.r]>id[b.r]):id[a.l]<id[b.l]; }
}q[MAX*10];
struct BLOCK
{
inline void query(int p,int a,int b)
{
int p1 = id[a],p2 = id[b];
//printf("begin: az=%d\n",p);
if(p1==p2)
{
//printf("doit id=%d\n",p);
for(re int i=a; i<=b; i++)
{
if(cnt1[i])
{ ans1[p] += cnt1[i],ans2[p]++; }
}
return ;
}
//printf("doit id=%d\n",p);
for(re int i=a; id[i]==p1; i++)
{
if(cnt1[i])
{ ans1[p] += cnt1[i],ans2[p]++; }
}
for(re int i=p1+1; i<=p2-1; i++)
{ ans1[p] += cnt2[i],ans2[p] += cnt3[i]; }
for(re int i=b; id[i]==p2; i--)
{
if(cnt1[i])
{ ans1[p] += cnt1[i],ans2[p]++; }
}
}
}block;
inline void add(int p)
{
cnt1[val[p]]++;
cnt2[id[val[p]]]++;
if(cnt1[val[p]]==1)
{ cnt3[id[val[p]]]++; }
}
inline void del(int p)
{
cnt1[val[p]]--;
cnt2[id[val[p]]]--;
if(cnt1[val[p]]==0)
{ cnt3[id[val[p]]]--; }
}
signed main()
{
cin >> n >> m; len = sqrt(n);
for(re int i=1; i<=n; i++)
{ cin >> val[i]; id[i] = (i-1)/len+1; }
for(re int i=1,l,r,a,b; i<=m; i++)
{ q[i] = (mo_team){(cin >> l,l),(cin >> r,r),(cin >> a,a),(cin >> b,b),i}; }
sort(q+1,q+1+m);
int l = 1,r = 0;
for(re int i=1,ql,qr; i<=m; i++)
{
ql = q[i].l,qr = q[i].r;
//printf("FUCK\n");
while(l>ql)
{ add(--l); }
while(r<qr)
{ add(++r); }
while(l<ql)
{ del(l++); }
while(r>qr)
{ del(r--); }
block.query(q[i].p,q[i].a,q[i].b);
}
for(re int i=1; i<=m; i++)
{ printf("%d %d\n",ans1[i],ans2[i]); }
return 0;
}
}
signed main()
{ return OMA::main(); }
其他
其实是不知道该放在哪里好
首先考虑暴力如何写,肯定是从当前这个位置一直往后跳,直到跳出序列,最劣 \(O(nm)\) 显然不可做。
考虑如何来优化。
因为这里是分块,所以我们考虑用分块的思想来解决这个问题
我们可以预处理出当前点跳出其所在块所需的最少步数和这个点跳出当前块后,所跳到的点,第 \(i\) 个点是由 第\(i+1\) 个点转移过来的,所以倒序来做,\(O(n)\) 即可解决。
查询直接一块一块跳,修改时只会影响到当前块,所以只更新修改点所在块即可,因为其他点在跳到修改后的块的时候,再跳的点也是修改后的,所以不用修改所有块。两个操作都是 \(O(\sqrt{n})\) 的。
总复杂度 \(O(m\sqrt{n})\) 。
然后一个非常草蛋sb的坑点就是装置位置是从零开始的。
Code
#include<cmath>
#include<cstdio>
#define MAX 200020
const int LEN = 450;
#define re register
namespace some
{
struct stream
{
template<typename type>inline stream &operator >>(type &s)
{
int w=1; s=0; char ch=getchar();
while(ch<'0'||ch>'9'){ if(ch=='-')w=-1; ch=getchar(); }
while(ch>='0'&&ch<='9'){ s=s*10+ch-'0'; ch=getchar(); }
return s*=w,*this;
}
}cin;
inline int min(int a,int b)
{ return a<b?a:b; }
}using namespace some;
namespace OMA
{
int n,m,k[MAX];
int len,id[MAX];
int l[LEN],r[LEN];
int to[MAX],sum[MAX];
struct BLOCK
{
inline void update(int p,int val)
{
k[p] = val;
for(re int i=r[id[p]]; i>=l[id[p]]; i--)
{
to[i] = i+k[i];
if(to[i]>r[id[p]])
{ sum[i] = 1; }
else
{
sum[i] = sum[to[i]]+1;
to[i] = to[to[i]];
}
}
}
inline int query(int p)
{
int res = 0;
while(p<=n)
{ res += sum[p],p = to[p]; }
return res;
}
}block;
signed main()
{
cin >> n; len = sqrt(n);
for(re int i=1; i<=n; i++)
{ cin >> k[i]; id[i] = (i-1)/len+1; }
for(re int i=1; i<=id[n]; i++)
{ l[i] = r[i-1]+1,r[i] = min(i*len,n); }
for(re int i=n; i; i--)
{
to[i] = i+k[i];
if(to[i]>r[id[i]])
{ sum[i] = 1; }
else
{
sum[i] = sum[to[i]]+1;
to[i] = to[to[i]];
}
}
cin >> m;
for(re int i=1,opt,pos,val; i<=m; i++)
{
cin >> opt >> pos,pos++;
if(opt==1)
{ printf("%d\n",block.query(pos)); }
else
{ block.update(pos,(cin >> val,val)); }
}
return 0;
}
}
signed main()
{ return OMA::main(); }