浅谈分块:https://www.cnblogs.com/AKMer/p/10369816.html

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

第一次的答案可以直接用树状数组求。

如果交换\(pos_1\)和\(pos_2\),那么显然我不需要管\([1,pos_1-1]\)和\([pos_2+1,n]\)。

对于\([pos_1+1,pos_2-1]\)之间的每个数\(v_i\)

\(v_i<v_{pos_1}\),答案减一;\(v_i>v_{pos_1}\),答案加一;\(v_i<v_{pos_2}\),答案加一,\(v_i>v_{pos_2}\),答案减一。

对于每个块我用一个树状数组维护块内权值个数。整个的块直接查找有多少小于或者大于某个值的数的个数,零散的直接暴力扫。

时间复杂度:\(O(NlogN+M\sqrt{N}logN)\)

空间复杂度:\(O(N\sqrt{n})\)

代码如下:

#include <cmath>
#include <cstdio>
#include <algorithm>
using namespace std;
#define low(i) ((i)&(-(i))) const int maxn=2e4+5; int L[145],R[145];
int n,m,cnt,block,ans;
int tmp[maxn],v[maxn],bel[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 Tree_array {
int c[maxn]; void add(int pos,int num) {
for(int i=pos;i<=cnt;i+=low(i))
c[i]+=num;
} int query(int pos) {
int res=0;
for(int i=pos;i;i-=low(i))
res+=c[i];
return res;
}
}T[145]; void check(int i,int l,int r) {
if(v[i]<v[l])ans--;
if(v[i]>v[l])ans++;
if(v[i]<v[r])ans++;
if(v[i]>v[r])ans--;
} int main() {
n=read(),block=sqrt(n);
for(int i=1;i<=n;i++) {
v[i]=tmp[i]=read(),bel[i]=(i-1)/block+1;
if(bel[i]!=bel[i-1])R[bel[i-1]]=i-1,L[bel[i]]=i;
}R[bel[n]]=n;
sort(tmp+1,tmp+n+1);
cnt=unique(tmp+1,tmp+n+1)-tmp-1;
for(int i=1;i<=n;i++)
v[i]=lower_bound(tmp+1,tmp+cnt+1,v[i])-tmp;
for(int i=n;i;i--) {
ans+=T[0].query(v[i]-1);
T[0].add(v[i],1);
T[bel[i]].add(v[i],1);
}
printf("%d\n",ans);
m=read();
while(m--) {
int l=read(),r=read();
if(r<l)swap(l,r);
if(bel[l]==bel[r]) {
for(int i=l+1;i<r;i++)
check(i,l,r);
}
else {
for(int i=l+1;i<=R[bel[l]];i++)
check(i,l,r);
for(int i=L[bel[r]];i<r;i++)
check(i,l,r);
for(int i=bel[l]+1;i<bel[r];i++) {
ans-=T[i].query(v[l]-1);
ans+=T[i].query(cnt)-T[i].query(v[l]);
ans+=T[i].query(v[r]-1);
ans-=T[i].query(cnt)-T[i].query(v[r]);
}
T[bel[l]].add(v[l],-1),T[bel[l]].add(v[r],1);
T[bel[r]].add(v[l],1),T[bel[r]].add(v[r],-1);
}
if(v[l]>v[r])ans--;
if(v[l]<v[r])ans++;
swap(v[l],v[r]);
printf("%d\n",ans);
}
return 0;
}

最新文章

  1. Android中的 init.rc文件简介
  2. UIDynamic--动力元素行为:UIDynamicItemBehavior
  3. 实验记录:Oracle redo logfile的resize过程
  4. size_t和size_type
  5. JS操作cookies方法
  6. Python HeapSort
  7. Error in notifier
  8. Cannot start service SPUserCodeV4 on computer
  9. Java操作Oracle数据库以及调用存储过程
  10. WPF学习之绘图和动画--DarrenF
  11. 走向DBA[MSSQL篇] 详解游标
  12. Ubuntu系统桌面任务栏和启动器全部消失解决方案
  13. python学习笔记(十 四)、web.py
  14. 三十三、ajaxFileUpload图片上传
  15. 2.4 random 模块
  16. Scrapy 解决Scrapy安装时报错&quot;Microsoft Visual C++ 14.0 is required&quot;
  17. C++ ifstream ofstream
  18. html5移动端查找
  19. 牛客小白月赛7 B 自杀游戏
  20. Java 容器之 Connection栈队列及一些常用

热门文章

  1. level-8
  2. [SCOI2005]超级格雷码
  3. NAS、SAN、DAS 说明
  4. 【转载】Android端百度地图API使用详解
  5. springmvc 学习笔记1
  6. shell 计算文件交并差
  7. android.intent.category.LAUNCHER和android.intent.action.MAIN
  8. 【转】一次完整的HTTP请求所经历的7个步骤
  9. BZOJ3243/UOJ121 [Noi2013]向量内积
  10. Zabbix的基本安装配置