题目链接

题意分析

我们考虑 交换两个数\([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 [国家集训队]排队的更多相关文章

  1. P1975 [国家集训队]排队 线段树套平衡树维护动态逆序对查询

    $ \color{#0066ff}{ 题目描述 }$ 排排坐,吃果果,生果甜嗦嗦,大家笑呵呵.你一个,我一个,大的分给你,小的留给我,吃完果果唱支歌,大家乐和和. 红星幼儿园的小朋友们排起了长长地队伍 ...

  2. 洛谷 P1975 [国家集训队]排队 Lebal:块内排序+树状数组

    题目描述 排排坐,吃果果,生果甜嗦嗦,大家笑呵呵.你一个,我一个,大的分给你,小的留给我,吃完果果唱支歌,大家乐和和. 红星幼儿园的小朋友们排起了长长地队伍,准备吃果果.不过因为小朋友们的身高有所区别 ...

  3. 「Luogu P1975 [国家集训队]排队」

    题目大意 给出一个序列 \(h\),支持交换其中的两数,求出每一时刻的逆序对个数. 分析 求逆序对是 \(O(N\log_2N)\) 的,有 \(M\) 个操作,如果暴力求的话时间复杂度就是 \(O( ...

  4. 【LG1975】[国家集训队]排队

    [LG1975][国家集训队]排队 题面 洛谷 题解 又是一个偏序问题 显然\(CDQ\) 交换操作不好弄怎么办? 可以看成两次删除两次插入 排序问题要注意一下 代码 #include <ios ...

  5. Luogu-1975 [国家集训队]排队

    Luogu-1975 [国家集训队]排队 题面 Luogu-1975 题解 题意:给出一个长度为n的数列以及m个交换两个数的操作,问每次操作后逆序对数量 时间,下标和数的大小三维偏序,,,把交换操作看 ...

  6. [国家集训队]排队 [cdq分治]

    题面 洛谷 和动态逆序对那道题没有什么区别 把一个交换换成两个删除和两个插入 #include <cstdio> #include <cstdlib> #include < ...

  7. luogu1975 [国家集训队]排队

    思路 序列中 |i | 1| 2| 3| 4| 5| 6| 7| 8| 9| 10| |----|--|--|--|--|--|--|--|--|--|--| |a[i]| a| b| c| L| d ...

  8. [luoguP1975] [国家集训队]排队(分块)

    传送门 直接暴力分块,然后在每一个块内排序. 查询时可以在每一个块内二分. #include <cmath> #include <cstdio> #include <io ...

  9. BZOJ 2039: [2009国家集训队]employ人员雇佣

    2039: [2009国家集训队]employ人员雇佣 Time Limit: 20 Sec  Memory Limit: 259 MBSubmit: 1369  Solved: 667[Submit ...

随机推荐

  1. 给tabhost加上点击监听,不是onTabChanged(String)监听

    给tabhost加上点击监听,不是onTabChanged(String)监听 2012-08-11 01:43 5209人阅读 评论(0) 收藏 举报 stringandroidlayoutnull ...

  2. 用WORD2007发布博客文章

    目前大部分的博客作者在用Word写博客这件事情上都会遇到以下3个痛点: 1.所有博客平台关闭了文档发布接口,用户无法使用Word,Windows Live Writer等工具来发布博客.使用Word写 ...

  3. 主键冲突异常 DuplicateKeyException

    org.springframework.dao.DuplicateKeyException Exception thrown when an attempt to insert or update d ...

  4. struts2 动态Action

    1.java 2.struts.xml struts2.5,默认关闭动态Action,着色的是开启和使用动态action 3.JSP 小结:访问时,用!后跟方法名的方法,方法返回值----->r ...

  5. POJ1269 Intersecting Lines 2017-04-16 19:43 50人阅读 评论(0) 收藏

    Intersecting Lines Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 15478   Accepted: 67 ...

  6. [label][转载][paypal]paypal在线支付接口的WEB语言设置

    http://stephen830.iteye.com/blog/274072 ★★★ 本篇为原创,需要引用转载的朋友请注明:< http://stephen830.iteye.com/blog ...

  7. Android Camera的使用(一) 读书笔记

    原文地址 https://blog.csdn.net/junzia/article/details/52301199 拍照步骤1.添加权限2.开启相机时check一下是否有摄像头3.对预览大小.照片大 ...

  8. Spring AOP 自定义注解获取http接口及WebService接口入参和出参

    注解方法实现过程中可以采用如下获取方式:—以下为例  HttpServletRequest request = ((ServletRequestAttributes) RequestContextHo ...

  9. java 传参数时 类型后跟 3个点 "..." 的意义

    对照代码和运行结果便知"..." 的意义 import java.util.ArrayList; public class StringDemo { public static v ...

  10. mysql in语句在java中的使用

    做权限的时候,通过role角色中的roleid,在auth权限中查找角色对应的权限. sql语句: SELECT authName from auth where authId in (1,2,3,4 ...