AT_agc030_d [AGC030D] Inversion Sum题解

AT_agc030_d [AGC030D] Inversion Sum

题目描述

给定一个长度为 NNN 的整数序列 A1,A2,…,ANA_1,A_2,\ldots,A_NA1,A2,,AN。你需要依次进行 QQQ 次操作。第 iii 次操作由两个整数 Xi,YiX_i,Y_iXi,Yi 给出,你可以从以下两种操作中恰好选择一种执行:

  • 交换 AXiA_{X_i}AXiAYiA_{Y_i}AYi 的值;
  • 什么都不做。

所有操作的执行方式共有 2Q2^Q2Q 种。请你求出对于所有可能的操作方式,最终得到的序列的逆序对数之和,并对 109+710^9+7109+7 取模。

其中,序列 P1,P2,…,PMP_1,P_2,\ldots,P_MP1,P2,,PM 的逆序对数定义为满足 1≤i<j≤M1\leq i<j\leq M1i<jMPi>PjP_i>P_jPi>Pj 的整数对 (i,j)(i,j)(i,j) 的个数。

输入格式

输入从标准输入读入,格式如下:

NNN QQQ
A1A_1A1 A2A_2A2 …\ldots ANA_NAN
X1X_1X1 Y1Y_1Y1
⋮\vdots
XQX_QXQ YQY_QYQ

输出格式

输出所有可能的最终序列的逆序对数之和对 109+710^9+7109+7 取模的结果。

输入输出样例 #1

输入 #1

3 2
1
2
3
1 2
1 3

输出 #1

6

输入输出样例 #2

输入 #2

5 3
3
2
3
1
4
1 5
2 3
4 2

输出 #2

36

输入输出样例 #3

输入 #3

9 5
3
1
4
1
5
9
2
6
5
3 5
8 9
7 9
3 2
3 8

输出 #3

425

说明/提示

限制条件

  • 1≤N≤30001\leq N\leq 30001N3000
  • 0≤Q≤30000\leq Q\leq 30000Q3000
  • 0≤Ai≤109 (1≤i≤N)0\leq A_i\leq 10^9\ (1\leq i\leq N)0Ai109 (1iN)
  • 1≤Xi,Yi≤N (1≤i≤Q)1\leq X_i,Y_i\leq N\ (1\leq i\leq Q)1Xi,YiN (1iQ)
  • Xi≠Yi (1≤i≤Q)X_i\neq Y_i\ (1\leq i\leq Q)Xi=Yi (1iQ)
  • 输入均为整数

样例说明 1

所有可能的操作方式如下,共有 444 种:

  • 111 次和第 222 次都什么都不做。最终序列为 1,2,31,2,31,2,3,逆序对数为 000
  • 111 次什么都不做,第 222 次交换。最终序列为 3,2,13,2,13,2,1,逆序对数为 333
  • 111 次交换,第 222 次什么都不做。最终序列为 2,1,32,1,32,1,3,逆序对数为 111
  • 111 次和第 222 次都交换。最终序列为 3,1,23,1,23,1,2,逆序对数为 222

这几种情况下逆序对数之和为 0+3+1+2=60+3+1+2=60+3+1+2=6,输出 666

由 ChatGPT 4.1 翻译

思路

直接DP,设fi,jf_{i,j}fi,j为i<j概率,然后求和即可。

代码见下

#include<bits/stdc++.h> 
using namespace std;
long long n,mod=1e9+7,a[3005],q,x[3005],y[3005],f[3005][3005],g[3005][3005],op=0,op2=1;
int main(){
	cin>>n>>q;
    for(int i=1;i<=n;i++){
        cin>>a[i];
    }
    for(int i=1;i<=q;i++){
        cin>>x[i]>>y[i];
    }
    for(int i=1;i<=n;i++){
        for(int j=1;j<=n;j++){
            if(a[i]<a[j]){
                f[i][j]=1;
            }
        }
    }
    for(int i=1;i<=q;i++){
        op2=op2*2ll%mod;
        f[x[i]][y[i]]=f[y[i]][x[i]]=(f[x[i]][y[i]]+f[y[i]][x[i]])*500000004ll%mod;
        for(int j=1;j<=n;j++){
            if(j!=y[i]&&j!=x[i]){
                f[x[i]][j]=f[y[i]][j]=(f[x[i]][j]+f[y[i]][j])*500000004ll%mod;
                f[j][x[i]]=f[j][y[i]]=(f[j][x[i]]+f[j][y[i]])*500000004ll%mod;                
            }
        }
    }
    for(int i=1;i<=n;i++){
        for(int j=1;j<=i-1;j++){
            op=(op+f[i][j])%mod;
        }
    }
    cout<<op*op2%mod<<endl;
	return 0; 	
}
posted @ 2025-12-11 21:24  bz02_2023f2  阅读(1)  评论(0)    收藏  举报  来源