BZOJ3787:Gty的文艺妹子序列(分块,树状数组)
Description
Input
Output
对每个询问,单独输出一行,表示al...ar中的逆序对数。对每个询问,单独输出一行,表示al...ar中的逆序对数。
Sample Input
1 7 5 6 9 4 9 4 4 7
10
0 4 6
0 5 8
0 1 10
1 25 19
0 19 25
1 14 4
0 12 12
0 2 5
1 8 7
1 1 10
Sample Output
2
3
16
13
0
2
Solution
分块,同时维护几个数组:
$f[i]$表示第$i$块内部的逆序对个数。
$g[i][j]$表示第$i$块和第$j$块能形成的逆序对个数,第二维用树状数组维护。
$s[i][j]$表示前$i$块里$j$数出现的次数,第二维同样用树状数组维护。
查询就利用这三个数组查一下,利用好树状数组前缀和的性质。乱七八糟加加减减。
修改就维护好这三个数组就好了……
速度倒数第二$46s$低空飞过……心情简单……
Code
#include<iostream>
#include<cstring>
#include<cstdio>
#include<cmath>
#define N (50009)
#define S (259)
using namespace std; int n,m,opt,l,r,ans,num,f[S],a[N];
int ID[N],L[S],R[S]; inline int read()
{
int x=,w=; char c=getchar();
while (c<'' || c>'') {if (c=='-') w=-; c=getchar();}
while (c>='' && c<='') x=x*+c-'', c=getchar();
return x*w;
} struct BIT
{
int c[N];
void Update(int x,int v)
{
for (; x<=n; c[x]+=v,x+=(x&-x));
}
int Query(int x)
{
int ans=;
for (; x; ans+=c[x],x-=(x&-x));
return ans;
}
}B,g[S],s[S]; void Build()
{
int unit=sqrt(n);
num=n/unit+(n%unit!=);
for (int i=; i<=num; ++i)
L[i]=(i-)*unit+, R[i]=i*unit;
R[num]=n;
for (int i=; i<=num; ++i)
for (int j=L[i]; j<=R[i]; ++j) ID[j]=i;
} void Preprocess()
{
for (int i=; i<=num; ++i)
{
for (int j=L[i]; j<=R[i]; ++j)
{
f[i]+=B.Query(n)-B.Query(a[j]);
B.Update(a[j],);
}
for (int j=L[i]; j<=R[i]; ++j) B.Update(a[j],-);
}
for (int i=; i<=num; ++i)
for (int j=; j<=i-; ++j)
{
for (int k=L[j]; k<=R[j]; ++k) B.Update(a[k],);
for (int k=L[i]; k<=R[i]; ++k) g[i].Update(j,B.Query(n)-B.Query(a[k]));
for (int k=L[j]; k<=R[j]; ++k) B.Update(a[k],-);
}
for (int i=; i<=num; ++i)
for (int j=L[i]; j<=R[i]; ++j) s[i].Update(a[j],);
for (int i=; i<=num; ++i)
for (int j=; j<=n; ++j) s[i].Update(j,s[i-].Query(j)-s[i-].Query(j-));
} int Calc(int l,int r)
{
int ans=;
if (ID[l]==ID[r])
{
for (int i=l; i<=r; ++i)
{
ans+=B.Query(n)-B.Query(a[i]);
B.Update(a[i],);
}
for (int i=l; i<=r; ++i) B.Update(a[i],-);
return ans;
}
for (int i=l; i<=R[ID[l]]; ++i)
{
ans+=s[ID[r]-].Query(a[i]-)-s[ID[l]].Query(a[i]-);
ans+=B.Query(n)-B.Query(a[i]);
B.Update(a[i],);
}
for (int i=l; i<=R[ID[l]]; ++i) B.Update(a[i],-); for (int i=L[ID[r]]; i<=r; ++i)
{
ans+=(s[ID[r]-].Query(n)-s[ID[r]-].Query(a[i]))-(s[ID[l]].Query(n)-s[ID[l]].Query(a[i]));
ans+=B.Query(n)-B.Query(a[i]);
B.Update(a[i],);
}
for (int i=L[ID[r]]; i<=r; ++i) B.Update(a[i],-); for (int i=ID[l]+; i<=ID[r]-; ++i) ans+=f[i];
for (int i=ID[l]+; i<=ID[r]-; ++i)
ans+=g[i].Query(i-)-g[i].Query(ID[l]);
for (int i=l; i<=R[ID[l]]; ++i) B.Update(a[i],);
for (int i=L[ID[r]]; i<=r; ++i) ans+=B.Query(n)-B.Query(a[i]);
for (int i=l; i<=R[ID[l]]; ++i) B.Update(a[i],-);
return ans;
} void Change(int x,int k)
{
int id=ID[x];
for (int i=id; i<=num; ++i) s[i].Update(a[x],-);
for (int i=id; i<=num; ++i) s[i].Update(k,); for (int i=; i<=id-; ++i)
{
g[id].Update(i,-s[i].Query(n)+s[i].Query(a[x])+s[i-].Query(n)-s[i-].Query(a[x]));
g[id].Update(i,s[i].Query(n)-s[i].Query(k)-s[i-].Query(n)+s[i-].Query(k));
}
for (int i=id+; i<=num; ++i)
{
g[i].Update(id,-s[i].Query(a[x]-)+s[i-].Query(a[x]-));
g[i].Update(id,s[i].Query(k-)-s[i-].Query(k-));
} f[id]=; a[x]=k;
for (int i=L[id]; i<=R[id]; ++i)
{
f[id]+=B.Query(n)-B.Query(a[i]);
B.Update(a[i],);
}
for (int i=L[id]; i<=R[id]; ++i) B.Update(a[i],-);
} int main()
{
n=read();
Build();
for (int i=; i<=n; ++i) a[i]=read();
Preprocess();
m=read();
while (m--)
{
opt=read(); l=read(); r=read();
l^=ans; r^=ans;
if (opt==) printf("%d\n",ans=Calc(l,r));
else Change(l,r);
}
}
BZOJ3787:Gty的文艺妹子序列(分块,树状数组)的更多相关文章
- BZOJ3787 gty的文艺妹子序列 【树状数组】【分块】
题目分析: 首先这种乱七八糟的题目就分块.然后考虑逆序对的统计. 一是块内的,二是块之间的,三是一个块内一个块外,四是都在块外. 令分块大小为$S$. 块内的容易维护,单次维护时间是$O(S)$. 块 ...
- BZOJ 3787: Gty的文艺妹子序列 [分块 树状数组!]
传送门 题意:单点修改,询问区间内逆序对数,强制在线 看到加了!就说明花了不少时间.... 如果和上题一样预处理信息,用$f[i][j]$表示块i到j的逆序对数 强行修改的话,每个修改最多会修改$(\ ...
- BZOJ 3787 Gty的文艺妹子序列(分块+树状数组+前缀和)
题意 给出n个数,要求支持单点修改和区间逆序对,强制在线. n,m<=50000 题解 和不带修改差不多,预处理出smaller[i][j]代表前i块小于j的数的数量,但不能用f[i][j]代表 ...
- 【bzoj3744】Gty的妹子序列 分块+树状数组+主席树
题目描述 我早已习惯你不在身边, 人间四月天 寂寞断了弦. 回望身后蓝天, 跟再见说再见…… 某天,蒟蒻Autumn发现了从 Gty的妹子树(bzoj3720) 上掉落下来了许多妹子,他发现 她们排成 ...
- 【BZOJ3744】Gty的妹子序列 分块+树状数组
[BZOJ3744]Gty的妹子序列 Description 我早已习惯你不在身边, 人间四月天 寂寞断了弦. 回望身后蓝天, 跟再见说再见…… 某天,蒟蒻Autumn发现了从 Gty的妹子树(bzo ...
- BZOJ 3744 Gty的妹子序列 (分块+树状数组+主席树)
题面传送门 题目大意:给你一个序列,多次询问,每次取出一段连续的子序列$[l,r]$,询问这段子序列的逆序对个数,强制在线 很熟悉的分块套路啊,和很多可持久化01Trie的题目类似,用分块预处理出贡献 ...
- BZOJ 3744 Gty的妹子序列 分块+树状数组
具体分析见 搬来大佬博客 时间复杂度 O(nnlogn)O(n\sqrt nlogn)O(nnlogn) CODE #include <cmath> #include <cctyp ...
- BZOJ3787 : Gty的文艺妹子序列
将序列分成$\sqrt{n}$块,预处理出每两块之间的逆序对数,以及ap[i]表示前i块内数字出现次数的树状数组 预处理:$O(n\sqrt{n}\log n)$ 修改时,ap[i]可以在$O(\sq ...
- 【分块】【树状数组】bzoj3787 Gty的文艺妹子序列
题解懒得自己写了,Orz一发wangxz神犇的: http://bakser.gitcafe.com/2014/12/04/bzoj3787-Gty%E7%9A%84%E6%96%87%E8%89%B ...
随机推荐
- DataGridview启用列重新排序属性的作用
DataGridview是winform中经常用的控件,今天来了解一下启用列重新排序属性的作用 默认没有选中是不能改变列前后顺序的 启用列重新排序后如下图可以拖动列标题(列标头)来改变列前后顺序
- JSP与Servlet之间传值的一般方法,适合新手做练习
#JSP与Servlet之间传值 Jsp与Servlet之间的传值有两种,一种是Jsp传值给Sevlet,另一种是Servlet传值给Jsp:使用request.response对象完成传值,具体实现 ...
- SQL Server优化查询
1. 首先要搞明白什么叫执行计划? 执行计划是数据库根据SQL语句和相关表的统计信息作出的一个查询方案,这个方案是由查询优化器自动分析产生的,比如一条SQL语句如果用来从一个 10万条记录的表中查1条 ...
- (六)彻底理解synchronized
1.sychronized简介 在学习知识之前,我们先来看一个现象 public class SynchronizedDemo implements Runnable { private static ...
- Entity Framework系列文章目录
Entity Framework系列文章目录Entity Framework系列文章目录Entity Framework系列文章目录Entity Framework系列文章目录
- JavaScript--事件绑定及深入(26)
// 事件绑定分为两种: // 一种是传统事件绑定(内联模型/脚本模型);上一章内容; // 一种是现代事件绑定(DOM2级模型);现代事件绑定在传统事件绑定基础上提供了更强大的功能; 一 传统事件绑 ...
- 为JavaScript正名--读你不知道的JavaScript(持续更新..)
你不知道的JavaScript上卷 JavaScript和Java的关系就像Carnival和Car的关系一样,八竿子打不着. JavaScript易上手,但由于其本身的特殊性,相比其他语言能真正掌握 ...
- LOJ#505. 「LibreOJ β Round」ZQC 的游戏(最大流)
题意 题目链接 Sol 首先把第一个人能吃掉的食物删掉 然后对每个人预处理出能吃到的食物,直接限流跑最大流就行了 判断一下最后的最大流是否等于重量和 注意一个非常恶心的地方是需要把除1外所有人都吃不到 ...
- 各类无次数限制的免费API接口整理
各类无次数限制的免费API接口整理,主要是聚合数据上和API Store上的一些,还有一些其他的. 聚合数据提供30大类,160种以上基础数据API服务,国内最大的基础数据API服务,下面就罗列一些免 ...
- Oracle常用方法备份
1.UNION ALL 和UNION.INTERSECT.MINUS 连接所有的查询结果.去掉重复的结果.操作返回查询结果中公共的行.在进行两个表格或者两个查询结果的时候,返回在第一个表格/查询结果中 ...