题目链接:http://61.187.179.132/JudgeOnline/problem.php?id=2141

题意:给出一个数列A,每次交换两个数的位置。输出交换后逆序对的个数。

思路:首先,对于交换位置x和y,对于区间[x+1,y-1]的数字,小于A[x]的要减去,大于A[x]的要加上,大于A[y]的要减去,小于A[y] 的要加上。然后A[x]<A[y]则答案加1,A[x]<A[y]答案要减去1。查找修改用树状数组套一个treap即可。

struct node
{
    int val,size,pri,L,R,cnt;
};
  
  
node a[N*300];
int e;
  
 
int newNode(int val)
{
    int x=++e;;
    a[x].val=val;
    a[x].size=1;
    a[x].cnt=1;
    a[x].L=a[x].R=0;
    a[x].pri=rand();
    return x;
}
 
void pushUp(int x)
{
    if(x==0) return;
    a[x].size=a[x].cnt+a[a[x].L].size+a[a[x].R].size;
}
 
void rotL(int &x)
{
    int y=a[x].R;
    a[x].R=a[y].L;
    a[y].L=x;
      
    pushUp(x);
    pushUp(y);
    x=y;
}
  
void rotR(int &x)
{
    int y=a[x].L;
    a[x].L=a[y].R;
    a[y].R=x;
      
    pushUp(x);
    pushUp(y);
    x=y;
}
  
void insert(int &k,int val)
{
    if(k==0) k=newNode(val);
    else if(val<a[k].val) 
    {
        insert(a[k].L,val);
        if(a[a[k].L].pri>a[k].pri) rotR(k);
    }
    else if(val>a[k].val)
    {
        insert(a[k].R,val);
        if(a[a[k].R].pri>a[k].pri) rotL(k);
    }
    else a[k].cnt++;
    pushUp(k);
}
  
void del(int val,int &k)
{
    if(k==0) return;
    else if(val<a[k].val) del(val,a[k].L);
    else if(val>a[k].val) del(val,a[k].R);
    else
    {
        a[k].cnt--;
        if(a[k].cnt<=0)
        {
            if(a[k].L==0&&a[k].R==0) k=0;
            else if(a[k].L==0) k=a[k].R;
            else if(a[k].R==0) k=a[k].L;
            else
            {
                if(a[a[k].L].pri<a[a[k].R].pri) rotL(k),del(val,a[k].L);
                else rotR(k),del(val,a[k].R);
            }
        }
    }
    pushUp(k);
}
  
int d[N];
  
int getCnt(int t,int x)
{
    if(t==0) return 0;
    if(a[t].val==x) 
    {
        return a[a[t].L].size+a[t].cnt;
    }
    if(a[t].val>x) return getCnt(a[t].L,x);
    return a[t].cnt+a[a[t].L].size+getCnt(a[t].R,x);
}
 
 
int n,m;
int A[N];
  
void Set(int x,int k)
{
    while(x<=n) 
    {
        insert(A[x],k);
        x+=x&-x;
    }
}
 
void erase(int x,int k)
{
    while(x<=n) 
    {
        del(k,A[x]);
        x+=x&-x;
    }
}
 
int get(int x,int k)
{
    int ans=0;
    while(x) 
    {
        ans+=getCnt(A[x],k);
        x-=x&-x;
    }
    return ans;
}
  
pair<int,int> cal(int L,int R,int x)
{
    i64 a=get(R,x)-get(L-1,x);
    i64 b=get(R,x-1)-get(L-1,x-1);
    return MP(a-b,b);

 
int p[N];
 
int find(int low,int high,int x)
{
    int M;
    while(low<=high)
    {
        M=(low+high)>>1;
        if(p[M]==x) return M;
        if(p[M]>x) high=M-1;
        else low=M+1;
    }
}
  
void getInt(int &x)
{
    char c=getchar();
    while(!isdigit(c)) c=getchar();
    x=0;
    while(isdigit(c)) x=x*10+c-'0',c=getchar();
}
  
int main()
{
    getInt(n);
    int i;
    FOR1(i,n) getInt(d[i]),p[i]=d[i];
    sort(p+1,p+n+1);
    int M=unique(p+1,p+n+1)-(p+1);
    FOR1(i,n) d[i]=find(1,M,d[i]);
    i64 sum=0;
    pair<int,int> temp;
    FOR1(i,n) 
    {
        Set(i,d[i]);
        temp=cal(1,i-1,d[i]);
        sum+=i-1-temp.first-temp.second;
    }
     
    PR(sum);
    RD(m);
    int x,y;
    while(m--)
    {
        getInt(x);
        getInt(y);
        if(x>y) swap(x,y);
        if(d[x]==d[y])
        {
            PR(sum);
            continue;
        }
         
        if(d[x]>d[y]) sum--;
        else sum++;
         
        if(y-x>1) 
        {
            temp=cal(x+1,y-1,d[x]);
            sum-=temp.second;
            sum+=y-x-1-temp.first-temp.second;
              
            temp=cal(x+1,y-1,d[y]);
            sum+=temp.second;
            sum-=y-x-1-temp.first-temp.second;
        }
         
        erase(x,d[x]);
        erase(y,d[y]);
        swap(d[x],d[y]);
        Set(x,d[x]);
        Set(y,d[y]);
         
        PR(sum);
    }
}

BZOJ 2141 排队(树状数组套treap)的更多相关文章

  1. BZOJ 2141 排队(树状数组套主席树)

    解法很多的题,可以块套树状数组,可以线段树套平衡树.我用的是树状数组套主席树. 题意:给出一段数列,m次操作,每次操作是交换两个位置的数,求每次操作后的逆序对数.(n,m<=2e4). 对于没有 ...

  2. BZOJ - 3295 动态逆序对 (树状数组套treap)

    题目链接 思路和bzoj2141差不多,不过这道题的数据更强一些,线段树套treapT了,树状数组套treap卡过~~ #include<bits/stdc++.h> using name ...

  3. BZOJ2141排队——树状数组套权值线段树(带修改的主席树)

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

  4. bzoj2141 树状数组套Treap树

    题目大意是在能够改变两个数的位置的情况下计算逆序对数 这因为是动态记录逆序对 本来单纯逆序对只要用树状数组计算即可,但这里因为更新,所以利用TReap树的删点和增加点来进行更新 大致是把每个树状数组所 ...

  5. HDU 5618 Jam's problem again (cdq分治+BIT 或 树状数组套Treap)

    题意:给n个点,求每一个点的满足 x y z 都小于等于它的其他点的个数. 析:三维的,第一维直接排序就好按下标来,第二维按值来,第三维用数状数组维即可. 代码如下: cdq 分治: #pragma ...

  6. BZOJ 1901 Zju2112 Dynamic Rankings ——树状数组套主席树

    [题目分析] BZOJ这个题目抄的挺霸气. 主席树是第一时间想到的,但是修改又很麻烦. 看了别人的题解,原来还是可以用均摊的思想,用树状数组套主席树. 学到了新的姿势,2333o(* ̄▽ ̄*)ブ [代 ...

  7. 【BZOJ】1901: Zju2112 Dynamic Rankings(区间第k小+树状数组套主席树)

    http://www.lydsy.com/JudgeOnline/problem.php?id=1901 首先还是吐槽时间,我在zoj交无限tle啊!!!!!!!!我一直以为是程序错了啊啊啊啊啊啊. ...

  8. [BZOJ 3196] 213平衡树 【线段树套set + 树状数组套线段树】

    题目链接:BZOJ - 3196 题目分析 区间Kth和区间Rank用树状数组套线段树实现,区间前驱后继用线段树套set实现. 为了节省空间,需要离线,先离散化,这样需要的数组大小可以小一些,可以卡过 ...

  9. [BZOJ 1901] Dynamic Rankings 【树状数组套线段树 || 线段树套线段树】

    题目链接:BZOJ - 1901 题目分析 树状数组套线段树或线段树套线段树都可以解决这道题. 第一层是区间,第二层是权值. 空间复杂度和时间复杂度均为 O(n log^2 n). 线段树比树状数组麻 ...

随机推荐

  1. spring中的BeanFactory与ApplicationContext的作用和区别

    BeanFactory 和ApplicationContext Bean 工厂(com.springframework.beans.factory.BeanFactory)是Spring 框架最核心的 ...

  2. Jmeter(十一)_针对响应信息不明确的接口做关联

    下午写一个新功能的接口脚本,遇到几个技术问题,现在将解决方案写出来 1:做接口关联的时候,发现接口响应没有可以利用的信息.如下图只返回了一个成功的标识,这样的接口如何与之关联? 通过抓包观察后续的修改 ...

  3. MySql关联子查询

    mysql有时候把子查询优化的很差,最差的情景就是 在where子句中使用in. -----<高性能mysql第二版>4.4.1

  4. 网站/IIS/Web/WCF服务 访问共享目录 映射 的解决方案

    目录 问题案例 原因分析 解决问题 总结 问题案例 环境: 电脑A:winform程序: 电脑B:部署了一个文件上传的WCF服务在IIS上.且该服务的配置文件中已经增加 <identity im ...

  5. InnoDB: Operating system error number 87 in a file operation. 错误87的解决方法

    InnoDB: Operating system error number 87 in a file operation. 错误87的解决方法 140628  8:10:48 [Note] Plugi ...

  6. 003-Nginx 设置Header 获取真实IP

    1.X-Forwarded-For的定义: X-Forwarded-For:简称XFF头,它代表客户端,也就是HTTP的请求端真实的IP,只有在通过了HTTP 代理或者负载均衡服务器时才会添加该项.它 ...

  7. SQL 4

    SQL WHERE 子句 WHERE 子句用于过滤记录. SQL WHERE 子句 WHERE 子句用于提取那些满足指定标准的记录. SQL WHERE 语法 SELECT column_name,c ...

  8. Py之pandas:dataframe学习【转载】

    转自:https://www.tutorialspoint.com/python_pandas/python_pandas_dataframe.htm 1.数据框4特性 列是不同类型的数据元素. 每列 ...

  9. angular $scope.$watch

    在$scope内置的所有函数中,用得最多的可能就是$watch 函数了.当你的数据模型中某一部分发生变化时,$watch函数可以向你发出通知. 你可以监控单个对象的属性,也可以监控需要经过计算的结果( ...

  10. Android APP安装后不在桌面显示图标的应用场景举例和实现方法

    最近在为公司做一款车联网的产品,由于公司本身擅长于汽车解码器的研发,所以该产品的诊断功能的实现除了使用目前市面上车联网产品中大量使用的OBD协议外,还会使用一些专车专用协议去实现一些特殊的诊断功能,如 ...