题面

传送门:[CQOI2011]动态逆序对


Solution

一开始我看到pty巨神写这套题的时候,第一眼还以为是个SB题:这不直接开倒车线段树统计就完成了吗?

然后冷静思考了一分钟,猛然发现单纯的线段树并不能解决这个问题,好像还要在外面再套上一颗树。

这就很shit了。你问我资磁不资磁树套树,我是不资磁的,树套树是暴力数据结构,我能资磁吗?

很不幸,昨天现实狠狠地打了我一脸:时间不够开新坑的,不切题又浑身难受,找了半天题,还是把这道题拉了出来(哈,真香)

不扯淡了,这题还是很显然的。

考虑开倒车,我们一个一个往里面加树,然后统计一下这个数能对当前的数列有多少贡献,贡献很容易想到:我们只需要找到在他后面比他小的数以及在他前面比他大的数就好。

然后本蒟蒻写了个蜜汁线段树套splay。

时间复杂度是$O(n*log^2n)$,空间复杂度为$O(n*logn)$。理论上应该能过

可惜现实非常苦感:

.....

那咋搞啊。

那我上个线段树套权值线段树吧

然后又码了半个小时。

时空复杂度均为$O(n*log^2n)$ (这明显要MLE啊,问题是题解蜜汁能过)

可惜现实依旧苦感:

难道,改数据了?

接着,我copy了一发题解,交上去,A掉了.......

到目前为止,我还是想不通为啥开同样的数组,他A了,我T了。难道说他外层套的树状数组可以有效减少空间的消耗?

想不通,还请个位dalao赐教。


Code (并不能A)

线段树套splay:

#include<iostream>
#include<cstdio>
using namespace std;
long long read()
{
long long x=0,f=1; char c=getchar();
while(!isdigit(c)){if(c=='-') f=-1;c=getchar();}
while(isdigit(c)){x=x*10+c-'0';c=getchar();}
return x*f;
}
const int N=100000+1000;
struct TreeInTree
{
#define mid ((now_l+now_r)>>1)
#define lson (now<<1)
#define rson (now<<1|1)
#define root son[r][1]
static const int M=N*25;
int fa[M],son[M][2],size[M],cnt[M],num[M],to;
inline void update(int x)
{
size[x]=size[son[x][0]]+size[son[x][1]]+cnt[x];
}
inline void rotate(int x,int type)
{
int y=fa[x],z=fa[y];
son[z][y==son[z][1]]=x,fa[x]=z;
son[y][!type]=son[x][type],fa[son[x][type]]=y;
son[x][type]=y,fa[y]=x;
update(y),update(x);
}
void splay(int x,int to)
{
while(fa[x]!=to)
{
if(x==son[fa[x]][fa[x]==son[fa[fa[x]]][1]] and fa[fa[x]]!=to)
rotate(fa[x],x==son[fa[x]][0]);
rotate(x,x==son[fa[x]][0]);
}
}
void Insert(int w,int r)
{
if(root==0)
{
root=++to,fa[root]=r;
num[root]=w,update(root);
return;
}
int now=root,last=root;
while(now!=0)
last=now,now=son[now][w>num[now]];
now=++to,fa[now]=last,son[last][w>num[last]]=now;
num[now]=w,update(now);
splay(now,r);
}
int Query1(int x,int r)
{
int now=root,ans=0;
while(now!=0)
{
if(num[now]>=x)
now=son[now][0];
else
{
if(num[now]>num[ans]) ans=now;
now=son[now][1];
}
}
if(ans==0) return 0;
splay(ans,r);
return size[son[ans][0]]+cnt[ans];
}
int Query2(int x,int r)
{
int now=root,ans=0;
num[0]=0x3f3f3f3f;
while(now!=0)
{
if(num[now]>x)
{
if(num[now]<num[ans]) ans=now;
now=son[now][0];
}
else
now=son[now][1];
}
num[0]=0;
if(ans==0) return 0;
splay(ans,r);
return size[son[ans][1]]+cnt[ans];
}
int t[N<<2];
void Build(int now,int now_l,int now_r)
{
t[now]=++to;
if(now_l==now_r) return;
Build(lson,now_l,mid);
Build(rson,mid+1,now_r);
}
inline void Insert2(int x,int w,int now,int now_l,int now_r)
{
Insert(w,t[now]);
if(now_l!=now_r)
{
if(x<=mid) Insert2(x,w,lson,now_l,mid);
else Insert2(x,w,rson,mid+1,now_r);
}
}
int Query3(int l,int r,int w,int type,int now,int now_l,int now_r)
{
if(now_l>=l and now_r<=r)
{
if(type==1) return Query1(w,t[now]);
else return Query2(w,t[now]);
}
int sum=0;
if(l<=mid) sum+=Query3(l,r,w,type,lson,now_l,mid);
if(r>mid) sum+=Query3(l,r,w,type,rson,mid+1,now_r);
return sum;
}
#undef mid
#undef lson
#undef rson
}tit;
int n,m,p[N],q[N],unOK[N];
long long ans[N];
int main()
{
freopen("3157.in","r",stdin);
freopen("3157.out","w",stdout); int t=clock();
n=read(),m=read();
for(int i=1;i<=n;i++)
p[read()]=i;
for(int i=1;i<=m;i++)
q[i]=read(),unOK[q[i]]=true; tit.Build(1,1,n);
for(int i=1;i<=n;i++)
if(unOK[i]==false)
{
tit.Insert2(p[i],i,1,1,n);
ans[m+1]+=tit.Query3(p[i],n,i,1,1,1,n)+tit.Query3(1,p[i],i,2,1,1,n);
}
for(int i=m;i>=1;i--)
{
tit.Insert2(p[q[i]],q[i],1,1,n);
ans[i]=ans[i+1]+tit.Query3(p[q[i]],n,q[i],1,1,1,n)+tit.Query3(1,p[q[i]],q[i],2,1,1,n);
} for(int i=1;i<=m;i++)
printf("%lld\n",ans[i]);
cerr<<clock()-t<<endl;
return 0;
}

线段树套权值线段树

#include<iostream>
#include<cstdio>
using namespace std;
long long read()
{
long long x=0,f=1; char c=getchar();
while(!isdigit(c)){if(c=='-') f=-1;c=getchar();}
while(isdigit(c)){x=x*10+c-'0';c=getchar();}
return x*f;
}
const int N=100000+1000;
int n,m,p[N],q[N],unOK[N];
long long ans[N];
struct TreeInTree
{
#define mid ((now_l+now_r)>>1)
#define lson (now<<1)
#define rson (now<<1|1)
static const int M=N*200;
int son[M][2],num[M],to;
inline void update(int x)
{
num[x]=num[son[x][0]]+num[son[x][1]];
}
void Insert(int x,int now,int now_l,int now_r)
{
if(now_l==now_r)
{
num[now]++;
return;
}
if(now>to)
cerr<<to;
if(x<=mid)
{
if(son[now][0]==0) son[now][0]=++to;
Insert(x,son[now][0],now_l,mid);
}
else
{
if(son[now][1]==0) son[now][1]=++to;
Insert(x,son[now][1],mid+1,now_r);
}
update(now);
}
int Query1(int l,int r,int now,int now_l,int now_r)
{
if(l>r) return 0;
if((now_l>=l and now_r<=r) or now==0)
return num[now];
int t_ans=0;
if(l<=mid) t_ans+=Query1(l,r,son[now][0],now_l,mid);
if(r>mid) t_ans+=Query1(l,r,son[now][1],mid+1,now_r);
return t_ans;
}
int t[N<<2];
void Build(int now,int now_l,int now_r)
{
t[now]=++to;
if(now_l==now_r) return;
Build(lson,now_l,mid);
Build(rson,mid+1,now_r);
}
inline void Insert2(int x,int w,int now,int now_l,int now_r)
{
Insert(w,t[now],1,n);
if(now_l!=now_r)
{
if(x<=mid) Insert2(x,w,lson,now_l,mid);
else Insert2(x,w,rson,mid+1,now_r);
}
}
int Query3(int l,int r,int w,int type,int now,int now_l,int now_r)
{
if(now_l>=l and now_r<=r)
{
if(type==1) return Query1(1,w-1,t[now],1,n);
else return Query1(w+1,n,t[now],1,n);
}
int sum=0;
if(l<=mid) sum+=Query3(l,r,w,type,lson,now_l,mid);
if(r>mid) sum+=Query3(l,r,w,type,rson,mid+1,now_r);
return sum;
}
#undef mid
#undef lson
#undef rson
}tit;
int main()
{
n=read(),m=read();
for(int i=1;i<=n;i++)
p[read()]=i;
for(int i=1;i<=m;i++)
q[i]=read(),unOK[q[i]]=true; tit.Build(1,1,n);
for(int i=1;i<=n;i++)
if(unOK[i]==false)
{
tit.Insert2(p[i],i,1,1,n);
ans[m+1]+=tit.Query3(p[i],n,i,1,1,1,n)+tit.Query3(1,p[i],i,2,1,1,n);
}
for(int i=m;i>=1;i--)
{
tit.Insert2(p[q[i]],q[i],1,1,n);
ans[i]=ans[i+1]+tit.Query3(p[q[i]],n,q[i],1,1,1,n)+tit.Query3(1,p[q[i]],q[i],2,1,1,n);
} for(int i=1;i<=m;i++)
printf("%lld\n",ans[i]);
return 0;
}

[Luogu P3157][CQOI2011]动态逆序对 (树套树)的更多相关文章

  1. luogu P3157 [CQOI2011]动态逆序对(CDQ分治)

    题目描述 对于序列A,它的逆序对数定义为满足i<j,且Ai>Aj的数对(i,j)的个数.给1到n的一个排列,按照某种顺序依次删除m个元素,你的任务是在每次删除一个元素之前统计整个序列的逆序 ...

  2. Luogu P3157 [CQOI2011]动态逆序对

    题目链接 \(Click\) \(Here\) 这个题有点卡常数..我的常数比较大所以是吸着氧气跑过去的... 题意:计算对于序列中每个位置\(p\),\([1,p-1]\)区间内比它大的数的个数,和 ...

  3. LUOGU P3157 [CQOI2011]动态逆序对(CDQ 分治)

    传送门 解题思路 cdq分治,将位置看做一维,修改时间看做一维,权值看做一维,然后就转化成了三维偏序,用排序+cdq+树状数组.注意算删除贡献时要做两次cdq,分别算对前面和后面的贡献. #inclu ...

  4. P3157 [CQOI2011]动态逆序对(树状数组套线段树)

    P3157 [CQOI2011]动态逆序对 树状数组套线段树 静态逆序对咋做?树状数组(别管归并QWQ) 然鹅动态的咋做? 我们考虑每次删除一个元素. 减去的就是与这个元素有关的逆序对数,介个可以预处 ...

  5. [BZOJ 3295] [luogu 3157] [CQOI2011]动态逆序对(树状数组套权值线段树)

    [BZOJ 3295] [luogu 3157] [CQOI2011] 动态逆序对 (树状数组套权值线段树) 题面 给出一个长度为n的排列,每次操作删除一个数,求每次操作前排列逆序对的个数 分析 每次 ...

  6. P3157 [CQOI2011]动态逆序对

    P3157 [CQOI2011]动态逆序对 https://www.luogu.org/problemnew/show/P3157 题目描述 对于序列A,它的逆序对数定义为满足i<j,且Ai&g ...

  7. 洛谷 P3157 [CQOI2011]动态逆序对 解题报告

    P3157 [CQOI2011]动态逆序对 题目描述 对于序列\(A\),它的逆序对数定义为满足\(i<j\),且\(A_i>A_j\)的数对\((i,j)\)的个数.给\(1\)到\(n ...

  8. P3157 [CQOI2011]动态逆序对 (CDQ解决三维偏序问题)

    P3157 [CQOI2011]动态逆序对 题目描述 对于序列A,它的逆序对数定义为满足i<j,且Ai>Aj的数对(i,j)的个数.给1到n的一个排列,按照某种顺序依次删除m个元素,你的任 ...

  9. BZOJ_3295_[Cqoi2011]动态逆序对_CDQ分治+树状数组

    BZOJ_3295_[Cqoi2011]动态逆序对_CDQ分治+树状数组 Description 对于序列A,它的逆序对数定义为满足i<j,且Ai>Aj的数对(i,j)的个数.给1到n的一 ...

随机推荐

  1. 【Python】数据结构

    列表的更多特性 list.append(x) 在列表的末尾添加一个元素.相当于 a[len(a):] = [x] . list.extend(iterable) 使用可迭代对象中的所有元素来扩展列表. ...

  2. 基于Huggingface使用BERT进行文本分类的fine-tuning

    随着BERT大火之后,很多BERT的变种,这里借用Huggingface工具来简单实现一个文本分类,从而进一步通过Huggingface来认识BERT的工程上的实现方法. 1.load data tr ...

  3. MacOS下Terminal获取GPS经纬度坐标

    通过命令行直接获取经纬度坐标MacOS 首先下载WhereAmI,最新版本: https://github.com/robmathers/WhereAmI/releases/download/v1.1 ...

  4. 开源后台系统*mee-admin*

    mee-admin开源后台系统 Preface 这是一个开放的时代,我们不能总是把东西揣在口袋里面自己乐呵. 也正如名言所说的"如果你有两块面包,你当用其中一块去换一朵水仙花" 所 ...

  5. JVM笔记五-堆区

    JVM笔记五-堆区 在JVM中,堆区是重中之重.通过前面文章的学习,我们知道了,栈区是不会有垃圾回收的,所以,经常说的垃圾回收,其实就是回收的是堆区的数据.在这里,我们将会看到传说中的,新生代.老年代 ...

  6. .NET 云原生架构师训练营(模块一 架构师与云原生)--学习笔记

    目录 什么是软件架构 软件架构的基本思路 单体向分布式演进.云原生.技术中台 1.1 什么是软件架构 1.1.1 什么是架构? Software architecture = {Elements, F ...

  7. 多测师_高级讲师肖sir讲解html中 Button跳转连接方法归纳

    第一种方法: 1.1<a href="http://www.baidu.com">   <input type="button" name=& ...

  8. 多测师讲解pthon_002字符,列表,元组,字段等

    # # # 索引:# # # 正向索引:  0  1  2  3  4  5  6# # #      l=   a  b  c  d  e  f  g# # # 反向索引: -7 -6 -5 -4 ...

  9. sop服务治理

    一,为什么需要服务治理: 我们最先接触的单体架构, 整个系统就只有一个工程, 打包往往是打成了 war 包, 然后部署到单一 tomcat 上面, 这种就是单体架构, 如图: 假如系统按照功能划分了, ...

  10. nginx安全:修改对外的服务软件名称并隐藏版本号(nginx1.18.0)

    一,为什么要隐藏nginx真实的软件名称? 1,nginx响应的Server头部都会携带上服务软件的名字和版本信息, 服务器软件的版本信息暴光在外部,很容易被黑客了解到,就通过相应版本的漏洞来攻击服务 ...