题意

给出n个数,要求支持单点修改和区间逆序对,强制在线。
n,m<=50000

题解

和不带修改差不多,预处理出smaller[i][j]代表前i块小于j的数的数量,但不能用f[i][j]代表第i块到第j块逆序对的数量,这样不好维护。

我们用f[i][j]代表从第i块选出一个元素与从第j块选出一个元素组成逆序对的数量,维护时最多修改根号n个f数组,查询时用前缀和起到与不带修改时f数组的作用。

其他部分和不带修改时差不多。

然后问题就解决了。

 #include<iostream>
#include<cstring>
#include<cstdio>
#include<cmath>
#include<algorithm>
using namespace std;
const int N=;
int n,Block,a[N],block[N],L[],R[],tr1[][],tr[][N],ans,m;
int lowbit(int x){
return x&-x;
}
void add(int id,int x,int w){
for(int i=x;i<=n;i+=lowbit(i)){
tr[id][i]+=w;
}
}
int getsum(int id,int x){
int tmp=;
for(int i=x;i;i-=lowbit(i)){
tmp+=tr[id][i];
}
return tmp;
}
void add1(int id,int x,int w){
for(int i=x;i<=block[n];i+=lowbit(i)){
tr1[id][i]+=w;
}
}
int getsum1(int id,int x){
int tmp=;
for(int i=x;i;i-=lowbit(i)){
tmp+=tr1[id][i];
}
return tmp;
}
int main(){
scanf("%d",&n);Block=sqrt(n);
for(int i=;i<=n;i++){
scanf("%d",&a[i]);
block[i]=(i-)/Block+;
if(!L[block[i]])L[block[i]]=i;
R[block[i]]=i;
}
for(int i=;i<=block[n];i++)
for(int j=L[i];j<=n;j++){
add(i,a[j],);
}
for(int i=;i<=block[n];i++){
for(int j=L[i];j<=R[i];j++){
add(,a[j],);
add1(i,i,getsum(,n)-getsum(,a[j]));
}
for(int j=R[i]+;j<=n;j++){
add1(i,block[j],getsum(,n)-getsum(,a[j]));
}
for(int j=L[i];j<=R[i];j++){
add(,a[j],-);
}
}
scanf("%d",&m);
while(m--){
int k,x,y;
scanf("%d%d%d",&k,&x,&y);
x^=ans;y^=ans;
if(k==){
for(int i=;i<=block[x]-;i++){
int size=R[i]-L[i]+;
add1(i,block[x],size-(getsum(i,y)-getsum(i+,y))-(size-(getsum(i,a[x])-getsum(i+,a[x]))));
}
for(int i=block[x]+;i<=block[n];i++){
add1(block[x],i,getsum(i,y-)-getsum(i+,y-)-(getsum(i,a[x]-)-getsum(i+,a[x]-)));
}
for(int i=L[block[x]];i<=R[block[x]];i++){
add(,a[i],);
add1(block[x],block[x],-(getsum(,n)-getsum(,a[i])));
}
for(int i=L[block[x]];i<=R[block[x]];i++){
add(,a[i],-);
}
for(int i=;i<=block[x];i++){
add(i,a[x],-);add(i,y,);
}
a[x]=y;
for(int i=L[block[x]];i<=R[block[x]];i++){
add(,a[i],);
add1(block[x],block[x],getsum(,n)-getsum(,a[i]));
}
for(int i=L[block[x]];i<=R[block[x]];i++){
add(,a[i],-);
}
}
else{
if(block[x]+>=block[y]){
ans=;
for(int i=x;i<=y;i++){
add(,a[i],);
ans+=getsum(,n)-getsum(,a[i]);
}
for(int i=x;i<=y;i++){
add(,a[i],-);
}
printf("%d\n",ans);
}
else{
ans=;
for(int i=block[x]+;i<=block[y]-;i++){
ans+=getsum1(i,block[y]-);
}
for(int i=x;i<=R[block[x]];i++){
add(,a[i],);
ans+=getsum(,n)-getsum(,a[i]);
ans+=getsum(block[x]+,a[i]-)-getsum(block[y],a[i]-);
}
for(int i=L[block[y]];i<=y;i++){
add(,a[i],);
ans+=getsum(,n)-getsum(,a[i]);
ans+=getsum(block[x]+,n)-getsum(block[y],n)-(getsum(block[x]+,a[i])-getsum(block[y],a[i]));
}
for(int i=x;i<=R[block[x]];i++){
add(,a[i],-);
}
for(int i=L[block[y]];i<=y;i++){
add(,a[i],-);
}
printf("%d\n",ans);
}
}
}
return ;
}

BZOJ 3787 Gty的文艺妹子序列(分块+树状数组+前缀和)的更多相关文章

  1. BZOJ 3787: Gty的文艺妹子序列 [分块 树状数组!]

    传送门 题意:单点修改,询问区间内逆序对数,强制在线 看到加了!就说明花了不少时间.... 如果和上题一样预处理信息,用$f[i][j]$表示块i到j的逆序对数 强行修改的话,每个修改最多会修改$(\ ...

  2. BZOJ3787:Gty的文艺妹子序列(分块,树状数组)

    Description Autumn终于会求区间逆序对了!Bakser神犇决定再考验一下他,他说道: “在Gty的妹子序列里,某个妹子的美丽度可也是会变化的呢.你还能求出某个区间中妹子们美丽度的逆序对 ...

  3. BZOJ3787 gty的文艺妹子序列 【树状数组】【分块】

    题目分析: 首先这种乱七八糟的题目就分块.然后考虑逆序对的统计. 一是块内的,二是块之间的,三是一个块内一个块外,四是都在块外. 令分块大小为$S$. 块内的容易维护,单次维护时间是$O(S)$. 块 ...

  4. BZOJ 3787: Gty的文艺妹子序列

    3787: Gty的文艺妹子序列 Time Limit: 50 Sec  Memory Limit: 256 MBSubmit: 186  Solved: 58[Submit][Status][Dis ...

  5. 【bzoj3744】Gty的妹子序列 分块+树状数组+主席树

    题目描述 我早已习惯你不在身边, 人间四月天 寂寞断了弦. 回望身后蓝天, 跟再见说再见…… 某天,蒟蒻Autumn发现了从 Gty的妹子树(bzoj3720) 上掉落下来了许多妹子,他发现 她们排成 ...

  6. BZOJ 3744 Gty的妹子序列 (分块+树状数组+主席树)

    题面传送门 题目大意:给你一个序列,多次询问,每次取出一段连续的子序列$[l,r]$,询问这段子序列的逆序对个数,强制在线 很熟悉的分块套路啊,和很多可持久化01Trie的题目类似,用分块预处理出贡献 ...

  7. 【BZOJ3744】Gty的妹子序列 分块+树状数组

    [BZOJ3744]Gty的妹子序列 Description 我早已习惯你不在身边, 人间四月天 寂寞断了弦. 回望身后蓝天, 跟再见说再见…… 某天,蒟蒻Autumn发现了从 Gty的妹子树(bzo ...

  8. BZOJ 3744 Gty的妹子序列 分块+树状数组

    具体分析见 搬来大佬博客 时间复杂度 O(nnlogn)O(n\sqrt nlogn)O(nn​logn) CODE #include <cmath> #include <cctyp ...

  9. BZOJ 4765 普通计算姬 dfs序+分块+树状数组+好题!!!

    真是道好题...感到灵魂的升华... 按dfs序建树状数组,拿前缀和去求解散块: 按点的标号分块,分成一个个区间,记录区间子树和 的 总和... 具体地,需要记录每个点u修改后,对每一个块i的贡献,记 ...

随机推荐

  1. sql中计算百分比

    sql中计算百分比:(转成字符串然后拼接%) ),) AS CHAR),'%') as aa from act_canal; 效果:

  2. Django(1.7 part1)

    django安装: django解压后目录下有一个setup.py文件,在命令行运行python setup.py install,当前前提是已经安装了python才能执行命令,然后用下面命令检查dj ...

  3. 命令行神器 cmder

    下载地址:http://cmder.net/ 修改命令提示符λ为$ 进入解压后的 cmder 的目录,进入 vendor,打开 clink.lua 文件. 修改 local cmder_prompt ...

  4. ModelDriven机制及其运用

    ModelDriven 为什么需要ModelDriven 所谓ModelDriven ,意思是直接把实体类当成页面数据的收集对象.比如,有实体类User 如下: package cn.com.lead ...

  5. express + jqPaginator 分页展示内容

    写在前面的话 分页展示内容也是我们在页面开发中经常会遇到的需求 前端页面利用jqPaginator这个jquery插件来编写 后端利用mysql存储数据 开始敲代码 回顾sql知识 首先让我们回顾一下 ...

  6. js的onclick和jq的click以及on和bind的区别

    onclick和click,只能静态绑定点击事件:bind的可以一次绑定多个事件(click/onmouseover等):on可以动态的绑定事件,当页面加载完成调用on即可

  7. [APIO2014]回文串(回文自动机)

    题意 给你一个由小写拉丁字母组成的字符串 s.我们定义 s 的一个子串的存在值为这个子串在 s 中出现的次数乘以这个子串的长度. 对于给你的这个字符串 s,求所有回文子串中的最大存在值. |S|< ...

  8. [洛谷P3982]龙盘雪峰信息解析器

    题目大意:给你一串代码,要求进行解码.解码规则详见题目. 解题思路:这是一道字符串处理的题目. 首先,有这么几种情况输出Error: 1.代码中出现除了0和1外的字符. 2.代码长度不是8的倍数. 3 ...

  9. ztree实现根节点单击事件,显示节点信息

    这段时间在维护公司的项目,去年做的项目里面有ztree树的例子,想起之前还没有开始写博客,一些知识点也无从找起,要新加一个右击节点事件,折腾了半天,其中也包含了一些知识点,稍稍做了一些demo. zT ...

  10. C实现头插法和尾插法来构建单链表(带头结点)

    我在之前一篇博客<C实现头插法和尾插法来构建单链表(不带头结点)>中具体实现了怎样使用头插法和尾插法来建立一个不带头结点的单链表,可是在实际使用中.我们用的最多的还是带头结点的单链表.今天 ...