P1975 [国家集训队]排队
题目链接
题意分析
我们考虑 交换两个数\([le,ri]\)的贡献
减少的逆序对数\([le,ri]\)中小于\(num[le]\)以及大于\(num[ri]\)的数
增加的\([le,ri]\)中大于\(num[le]\)以及小于\(num[ri]\)的数
同时注意 如果\(num[le]!=num[ri]\)
二者相互的贡献 就多算了一次
所以我们需要特判一下
至于修改和查询 我们可以使用树套树维护
CODE:
#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<cstdlib>
#include<string>
#include<queue>
#include<map>
#include<stack>
#include<list>
#include<set>
#include<deque>
#include<vector>
#include<ctime>
#define ll long long
#define inf 0x7fffffff
#define N 150008
#define IL inline
#define M 1008611
#define D double
#define ull unsigned long long
#define R register
using namespace std;
template<typename T>IL void read(T &_)
{
T __=0,___=1;char ____=getchar();
while(!isdigit(____)) {if(____=='-') ___=0;____=getchar();}
while(isdigit(____)) {__=(__<<1)+(__<<3)+____-'0';____=getchar();}
_=___ ? __:-__;
}
/*-------------OI使我快乐-------------*/
int n,q,tot,cnt,cnta,cntb,top;ll ans;
int num[N],res[N],sta[N],tmp[N];
int root[N],lson[N*50],rson[N*50],siz[N*50];
int cdy[N],wzy[N];
IL int new_code(){return top ? sta[top--]:++tot;}
IL void solve(int le,int ri)
{
if(le==ri) return;
int mid=(le+ri)>>1;
solve(le,mid);solve(mid+1,ri);
int ix=le,jx=mid+1,kx=le;
for(;ix<=mid&&jx<=ri;)
{
if(res[ix]<=res[jx]) tmp[kx++]=res[ix++];
else ans+=mid-ix+1,tmp[kx++]=res[jx++];
}
for(;ix<=mid;) tmp[kx++]=res[ix++];
for(;jx<=ri;) tmp[kx++]=res[jx++];
for(R int i=le;i<=ri;++i) res[i]=tmp[i];
}
IL void insert(int &now,int le,int ri,int pos,int d)
{
if(!now) now=new_code();siz[now]+=d;
if(le==ri) return;
int mid=(le+ri)>>1;
if(pos<=mid) insert(lson[now],le,mid,pos,d);
else insert(rson[now],mid+1,ri,pos,d);
if(!siz[now]) sta[++top]=now,now=0;
}
IL void add(int x,int pos,int d)
{
for(R int i=x;i<=n;i+=i&-i) insert(root[i],1,cnt,pos,d);
}
IL void prework(int le,int ri)
{
cnta=cntb=0;
for(R int x=le-1;x;x-=x&-x) cdy[++cnta]=root[x];
for(R int x=ri;x;x-=x&-x) wzy[++cntb]=root[x];
}
IL int getsum()
{
int res=0;
for(R int i=1;i<=cnta;++i) res-=siz[cdy[i]];
for(R int i=1;i<=cntb;++i) res+=siz[wzy[i]];
return res;
}
IL int getsumcdy()
{
int res=0;
for(R int i=1;i<=cnta;++i) res-=siz[lson[cdy[i]]];
for(R int i=1;i<=cntb;++i) res+=siz[lson[wzy[i]]];
return res;
}
IL int getsumwzy()
{
int res=0;
for(R int i=1;i<=cnta;++i) res-=siz[rson[cdy[i]]];
for(R int i=1;i<=cntb;++i) res+=siz[rson[wzy[i]]];
return res;
}
IL void pushcdy()
{
for(R int i=1;i<=cnta;++i) cdy[i]=lson[cdy[i]];
for(R int i=1;i<=cntb;++i) wzy[i]=lson[wzy[i]];
}
IL void pushwzy()
{
for(R int i=1;i<=cnta;++i) cdy[i]=rson[cdy[i]];
for(R int i=1;i<=cntb;++i) wzy[i]=rson[wzy[i]];
}
IL int get_cdy(int le,int ri,int pos)
{
if(pos<1) return 0;
if(le==ri) return getsum();
int mid=(le+ri)>>1,tmp=getsumcdy();
if(pos<=mid) {pushcdy();return get_cdy(le,mid,pos);}
else {pushwzy();return get_cdy(mid+1,ri,pos)+tmp;}
}
IL int get_wzy(int le,int ri,int pos)
{
if(pos>cnt) return 0;
if(le==ri) return getsum();
int mid=(le+ri)>>1,tmp=getsumwzy();
if(pos<=mid) {pushcdy();return get_wzy(le,mid,pos)+tmp;}
else {pushwzy();return get_wzy(mid+1,ri,pos);}
}
int main()
{
// freopen("cd.in","r",stdin);
// freopen("cd.out","w",stdout);
read(n);
for(R int i=1;i<=n;++i) read(num[i]),res[i]=num[i];
solve(1,n);cnt=unique(res+1,res+n+1)-res-1;
for(R int i=1;i<=n;++i) num[i]=lower_bound(res+1,res+cnt+1,num[i])-res;
for(R int i=1;i<=n;++i) add(i,num[i],1);
printf("%lld\n",ans);
// prework(1,n);
// for(R int i=1;i<=n;++i) printf("%d%c",num[i],(i==n ? '\n':' '));
// prework(1,n);
// printf("%d\n",get_cdy(1,cnt,cnt));
// prework(1,n);
// printf("%d\n",get_wzy(1,cnt,1));
read(q);
while(q--)
{
int x,y,tmpx,tmpy;read(x);read(y);
if(x==y) continue;tmpx=num[x];tmpy=num[y];
// printf("now is %d %d\n",tmpx,tmpy);
int cdy,wzy,zjz,ghj;
prework(x,y);
cdy=get_cdy(1,cnt,tmpx-1);
prework(x,y);
wzy=get_wzy(1,cnt,tmpx+1);
prework(x,y);
zjz=get_cdy(1,cnt,tmpy-1);
prework(x,y);
ghj=get_wzy(1,cnt,tmpy+1);
// printf("check %d %d %d %d\n",cdy,wzy,zjz,ghj);
ans=ans-cdy+wzy+zjz-ghj;
if(tmpx<tmpy) ans--;
else if(tmpx>tmpy) ans++;
add(x,tmpx,-1);add(y,tmpy,-1);
swap(num[x],num[y]);
add(x,tmpy,1);add(y,tmpx,1);
printf("%lld\n",ans);
}
// fclose(stdin);
// fclose(stdout);
return 0;
}
HEOI 2019 RP++
P1975 [国家集训队]排队的更多相关文章
- P1975 [国家集训队]排队 线段树套平衡树维护动态逆序对查询
$ \color{#0066ff}{ 题目描述 }$ 排排坐,吃果果,生果甜嗦嗦,大家笑呵呵.你一个,我一个,大的分给你,小的留给我,吃完果果唱支歌,大家乐和和. 红星幼儿园的小朋友们排起了长长地队伍 ...
- 洛谷 P1975 [国家集训队]排队 Lebal:块内排序+树状数组
题目描述 排排坐,吃果果,生果甜嗦嗦,大家笑呵呵.你一个,我一个,大的分给你,小的留给我,吃完果果唱支歌,大家乐和和. 红星幼儿园的小朋友们排起了长长地队伍,准备吃果果.不过因为小朋友们的身高有所区别 ...
- 「Luogu P1975 [国家集训队]排队」
题目大意 给出一个序列 \(h\),支持交换其中的两数,求出每一时刻的逆序对个数. 分析 求逆序对是 \(O(N\log_2N)\) 的,有 \(M\) 个操作,如果暴力求的话时间复杂度就是 \(O( ...
- 【LG1975】[国家集训队]排队
[LG1975][国家集训队]排队 题面 洛谷 题解 又是一个偏序问题 显然\(CDQ\) 交换操作不好弄怎么办? 可以看成两次删除两次插入 排序问题要注意一下 代码 #include <ios ...
- Luogu-1975 [国家集训队]排队
Luogu-1975 [国家集训队]排队 题面 Luogu-1975 题解 题意:给出一个长度为n的数列以及m个交换两个数的操作,问每次操作后逆序对数量 时间,下标和数的大小三维偏序,,,把交换操作看 ...
- [国家集训队]排队 [cdq分治]
题面 洛谷 和动态逆序对那道题没有什么区别 把一个交换换成两个删除和两个插入 #include <cstdio> #include <cstdlib> #include < ...
- luogu1975 [国家集训队]排队
思路 序列中 |i | 1| 2| 3| 4| 5| 6| 7| 8| 9| 10| |----|--|--|--|--|--|--|--|--|--|--| |a[i]| a| b| c| L| d ...
- [luoguP1975] [国家集训队]排队(分块)
传送门 直接暴力分块,然后在每一个块内排序. 查询时可以在每一个块内二分. #include <cmath> #include <cstdio> #include <io ...
- BZOJ 2039: [2009国家集训队]employ人员雇佣
2039: [2009国家集训队]employ人员雇佣 Time Limit: 20 Sec Memory Limit: 259 MBSubmit: 1369 Solved: 667[Submit ...
随机推荐
- IoC和DI的区别
------------------siwuxie095 IoC 和 DI 的区别 1.区别 (1)Io ...
- spring定时任务执行两次的原因与解决方法
spring定时任务,本地执行一次,放到服务器上后,每次执行时会执行两次,原因及解决办法. http://blog.csdn.net/yaobengen/article/details/7031266 ...
- Java 设计模式系列(一)单例模式
Java 设计模式系列(一)单例模式 保证一个类只有一个实例,并且提供一个访可该实例的全局访问点. 一.懒汉式单例 /** * 懒汉式单例类:在第一次调用的时候实例化自己 * 1. 构造器私有化,避免 ...
- 怎样application不被第三方应用杀掉--Android
方法: 对于放在/system/app下的应用,需要在其Manifest.xml文件中设置persistent属性,如应用程序'Phone'的AndroidManifest.xml文件: <ap ...
- Page Object页面设计模式核心要点
Page Object,页面对象.一种设计模式,实施selenium的最佳实践,体现了web应用与页面显示之间的关系.为什么需要Page Object?测试代码维护的需要:减少代码的编码量,减少代 ...
- Monocular Visual-Inertial Odometry单目视觉惯性里程计
Monocular Visual-Inertial Odometry:https://www.qualcomm.com/invention/research 单目视觉-惯性里程计 INDOOR POS ...
- ## 20155336 2016-2017-2《JAVA程序设计》第八周学习总结
20155336 2016-2017-2<JAVA程序设计>第八周学习总结 教材学习内容总结 第14章 NIO与NIO2 NIO简介 NIO使用频道来衔接数据结点,在处理数据时,NIO可以 ...
- 深入CSS属性(九):z-index
如果你不是一名csser新手,想必你对z-index的用法应该有个大致的了解了吧,z-index可以控制定位元素在垂直于显示屏方向(Z 轴)上的堆叠顺序,本文不去讲述基本的API如何使用,而是去更深入 ...
- Docker在github上的站点
https://github.com/docker (docker在github上的官方地址) https://github.com/dockerfile (docker官方镜像的Dockerfile ...
- asp.net mvc部分视图的action中获取父级视图信息
RouteData.DataTokens["ParentActionViewContext"]中包含了父级视图的相关信息,如路由等 public ActionResult Chil ...