bzoj3295 洛谷P3157、1393 动态逆序对——树套树
题目:bzoj3295 https://www.lydsy.com/JudgeOnline/problem.php?id=3295
洛谷 P3157(同一道题) https://www.luogu.org/problemnew/show/P3157
洛谷 P1393(略有不同) https://www.luogu.org/problemnew/show/P1393
动态逆序对问题;
树状数组套权值线段树,动态开点;
就像树状数组那样做就可以了,每个线段树维护一段区间内的不同权值的数的个数;
删除一个点,就把答案减去它贡献的逆序对数量就可以了;
有点不太懂空间范围怎么算...
代码如下:
#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
typedef long long ll;
int const maxn=1e5+,maxm=maxn*;
int n,m,pos[maxn],t[maxn],rt[maxn],ls[maxm],rs[maxm],sum[maxm],cnt;
ll ans;
ll getsum(int x)
{
ll ret=;
for(;x;x-=(x&-x)) ret+=t[x];
return ret;
}
void add(int x)
{
for(;x<=n;x+=(x&-x)) t[x]++;
}
void update(int &x,int l,int r,int p,int v)
{
if(!x)x=++cnt; sum[x]+=v;
if(l==r)return;
int mid=(l+r)>>;
if(p<=mid)update(ls[x],l,mid,p,v);
else update(rs[x],mid+,r,p,v);
}
void insert(int x,int p,int v)
{
for(;x<=n;x+=(x&-x)) update(rt[x],,n,p,v);
}
ll ask(int x,int l,int r,int p)
{
if(l==r)return sum[x];
int mid=(l+r)>>;
if(p<=mid)return ask(ls[x],l,mid,p);
return sum[ls[x]]+ask(rs[x],mid+,r,p);
}
ll query(int x,int p)//<=p 的个数
{
ll ret=;
for(;x;x-=(x&-x)) ret+=ask(rt[x],,n,p);
return ret;
}
ll pre(int x,int p){return query(p,n)-query(p,x);}
ll suf(int x,int p){return query(n,x)-query(p,x);}
int main()
{
scanf("%d%d",&n,&m);
for(int i=,x;i<=n;i++)
{
scanf("%d",&x); pos[x]=i;
ans+=getsum(n)-getsum(x);
add(x); insert(i,x,);
}
for(int i=,x;i<=m;i++)
{
printf("%lld\n",ans); scanf("%d",&x);
ans-=pre(x,pos[x])+suf(x,pos[x]);
insert(pos[x],x,-);
}
return ;
}
洛谷P1393,离散化一下即可。(通过数喜+1)
代码:
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
typedef long long ll;
int const maxn=4e4+,maxm=maxn*;
int n,m,t[maxn],rt[maxn],ls[maxm],rs[maxm],sum[maxm],cnt,tot;
ll ans,a[maxn],b[maxn];
ll getsum(int x)
{
ll ret=;
for(;x;x-=(x&-x)) ret+=t[x];
return ret;
}
void add(int x)
{
for(;x<=n;x+=(x&-x)) t[x]++;
}
void update(int &x,int l,int r,int p,int v)
{
if(!x)x=++cnt; sum[x]+=v;
if(l==r)return;
int mid=(l+r)>>;
if(p<=mid)update(ls[x],l,mid,p,v);
else update(rs[x],mid+,r,p,v);
}
void insert(int x,int p,int v)
{
for(;x<=n;x+=(x&-x)) update(rt[x],,n,p,v);
}
ll ask(int x,int l,int r,int p)
{
if(l==r)return sum[x];
int mid=(l+r)>>;
if(p<=mid)return ask(ls[x],l,mid,p);
return sum[ls[x]]+ask(rs[x],mid+,r,p);
}
ll query(int x,int p)//<=p 的个数
{
ll ret=;
for(;x;x-=(x&-x)) ret+=ask(rt[x],,n,p);
return ret;
}
ll pre(int x,int p){return query(p,n)-query(p,x);}
ll suf(int x,int p){return query(n,x)-query(p,x);}
int main()
{
scanf("%d%d",&n,&m);
for(int i=;i<=n;i++)scanf("%lld",&a[i]),b[i]=a[i];
sort(b+,b+n+);
tot=unique(b+,b+n+)-b-;
for(int i=;i<=n;i++) a[i]=lower_bound(b+,b+tot+,a[i])-b;
for(int i=;i<=n;i++)
{
ans+=getsum(n)-getsum(a[i]);
add(a[i]); insert(i,a[i],);
}
for(int i=,k;i<=m;i++)
{
printf("%lld ",ans); scanf("%d",&k);
ans-=pre(a[k],k)+suf(a[k],k);
insert(k,a[k],-);
}
printf("%lld\n",ans);
return ;
}
bzoj3295 洛谷P3157、1393 动态逆序对——树套树的更多相关文章
- 洛谷 P3157 [CQOI2011]动态逆序对 解题报告
P3157 [CQOI2011]动态逆序对 题目描述 对于序列\(A\),它的逆序对数定义为满足\(i<j\),且\(A_i>A_j\)的数对\((i,j)\)的个数.给\(1\)到\(n ...
- 洛谷 P3157 [CQOI2011]动态逆序对(树套树)
题面 luogu 题解 树套树(树状数组套动态开点线段树) 静态使用树状数组求逆序对就不多说了 用线段树代替树状数组,外面套树状数组统计每个点逆序对数量 设 \(t1[i]\)为\(i\)前面有多少个 ...
- 洛谷 P3157 [CQOI2011]动态逆序对 | CDQ分治
题目:https://www.luogu.org/problemnew/show/3157 题解: 1.对于静态的逆序对可以用树状数组做 2.我们为了方便可以把删除当成增加,可以化动为静 3.找到三维 ...
- 洛谷P3157 [CQOI2011]动态逆序对
题目大意: 给定\(1\)到\(n\)的一个排列,按照给定顺序依次删除\(m\)个元素,计算每个元素删除之前整个序列的逆序对数量 基本套路:删边变加边 那么我们不就是求满足\(pos_i<pos ...
- 【BZOJ3295】动态逆序对(线段树,树状数组)
[BZOJ3295]动态逆序对(线段树,树状数组) 题面 Description 对于序列A,它的逆序对数定义为满足iAj的数对(i,j)的个数.给1到n的一个排列,按照某种顺序依次删除m个元素,你的 ...
- P3157 [CQOI2011]动态逆序对(树状数组套线段树)
P3157 [CQOI2011]动态逆序对 树状数组套线段树 静态逆序对咋做?树状数组(别管归并QWQ) 然鹅动态的咋做? 我们考虑每次删除一个元素. 减去的就是与这个元素有关的逆序对数,介个可以预处 ...
- P3157 [CQOI2011]动态逆序对
P3157 [CQOI2011]动态逆序对 https://www.luogu.org/problemnew/show/P3157 题目描述 对于序列A,它的逆序对数定义为满足i<j,且Ai&g ...
- P3157 [CQOI2011]动态逆序对 (CDQ解决三维偏序问题)
P3157 [CQOI2011]动态逆序对 题目描述 对于序列A,它的逆序对数定义为满足i<j,且Ai>Aj的数对(i,j)的个数.给1到n的一个排列,按照某种顺序依次删除m个元素,你的任 ...
- BZOJ_3295_[Cqoi2011]动态逆序对_CDQ分治+树状数组
BZOJ_3295_[Cqoi2011]动态逆序对_CDQ分治+树状数组 Description 对于序列A,它的逆序对数定义为满足i<j,且Ai>Aj的数对(i,j)的个数.给1到n的一 ...
随机推荐
- python第一章计算机基础
第一章 计算机基础 1.1 硬件 计算机基本的硬件由:CPU / 内存 / 主板 / 硬盘 / 网卡 / 显卡 / 显示器 等组成,只有硬件但硬件之间无法进行交流和通信. 1.2 操作系统 操作系统用 ...
- python的学习之路(三)
一.set集合#!/usr/bin/env python# *_*coding:utf-8 *_*# Author: harson old_dict = { "#1": {'hos ...
- 关于微信小程序getUserInfo最新修改后,如何在原有项目的授权逻辑的调整
今天一大早调试小程序,结果出现这个...微信小程序也是醉了,这么大的改动,也没有通过开发者服务号通知一声 人在屋檐下不得不低头(改呗,那么如何以最小的代价更新呢,下面给我的解决方案) 原来我们在首次进 ...
- 牛客网NOIP赛前集训营 第6场 T1 最长路
[题解] 先建反向图,然后跑拓扑排序求出最长路. 将所有的点按照最长路从小到大分层,把上一层连向这一层的边按照边权为第一关键字.起点的排名为第二关键字排序. 按照这个顺序更新这一层的答案,按照这一层每 ...
- UVA 230 Borrowers (STL 行读入的处理 重载小于号)
题意: 输入若干书籍和作者名字,然后先按作者名字升序排列,再按标题升序排列,然后会有3种指令,BORROW,RETURN, SHELVE. BORROW 和 RETURN 都会带有一个书名在后面,如: ...
- Lucene实现全文检索的流程
[索引和搜索流程图] 对要索引的原始内容进行索引构建一个索引库,索引过程包括:确定原始内容即要搜索的内容->采集文档->创建文档->分析文档->索引文档. 从索引库中搜索内容, ...
- 如何把DEBIAN变成UBUNTU-DESKTOP最少化安装
Ubuntu 18.04没有32位,老电脑要装32位下面方法可以实现 Debian 9.6变成Ubuntu-Desktop 最少化 1 安装Debian 9.6时用expert mode 安装,安装过 ...
- mongodb & macOS
mongodb & macOS https://docs.mongodb.com/manual/tutorial/install-mongodb-on-os-x/ https://stacko ...
- [K/3Cloud] 关于单据转换的问题
1. 单据转换,是否支持重复下推,支持新增下推和更新下推? 答:支持重复下推,是否允许下推受以下因素: 1).源分录是否是有效状态(源单单头状态会自动影响分录,下同),例如已审核.未关闭.未作废: 2 ...
- SOJ 2930_积木城堡
[题意]若干个城堡,给定每个城堡的积木数及每块积木的棱长.从城堡中抽出积木使每块城堡高度相同,求最大高度 [分析]城堡的积木选择可以看成01背包问题,从最矮的城堡高度开始依次递减,求出使每个背包都能装 ...