51NOD 1020 逆序排列(dp预处理)

  定义dp[i][j]为1~i的排列里逆序数为j的方案数。那么只需要求i阶段的dp值时,只需要考虑在i-1阶段的基础上把i放哪里就好了,如果i放在倒数第v个就会新产生v-1个逆序对。那么转移方程就是dp[i][j]=sum(dp[i-1][j-v])  0<=v<=min (j,i-1),如果单单这样的话复杂度是n*k*k,因为转移是由上一阶段一个连续区间的dp值转移来的,所以在算i阶段dp值时可以先处理出i-1阶段dp值的前缀和,这样转移就O(1)了

#include<bits/stdc++.h>
using namespace std;
#define ls rt<<1
#define rs (rt<<1)+1
#define PI acos(-1)
#define eps 1e-8
#define ll long long
#define fuck(x) cout<<#x<<"     "<<x<<endl;
typedef pair<int,int> pii;
const int inf=2e9;
const int maxn=1e6+10;
const int mod=1e9+7;
int d[4][2]={1,0,-1,0,0,1,0,-1};
//int lowbit(int x){return x&-x;}
//void add(int x,int v){while(x<=n)bit[x]+=v,x+=lowbit(x);}
//int sum(int x){int ans=0;while(x>=1) ans+=bit[x],x-=lowbit(x);return ans;}
inline ll read() {
    ll s = 0,w = 1;
    char ch = getchar();
    while(!isdigit(ch)) {
        if(ch == '-') w = -1;
        ch = getchar();
    }
    while(isdigit(ch))
        s = s * 10 + ch - '0',ch = getchar();
    return s * w;
}
inline void write(ll x) {
    if(x < 0)
        putchar('-'), x = -x;
    if(x > 9)
        write(x / 10);
    putchar(x % 10 + '0');
}

int dp[1005][(int)2e4+10],pre[(int)2e4+10];

int main(){
    dp[1][1]=0,dp[1][0]=1;
    pre[0]=1,pre[1]=1;
    for(int i=2;i<=1000;i++){
        for(int j=0;j<=min(20000,i*(i-1)/2);j++) {
            if (i - 1 >= j)
                dp[i][j] = pre[j];
            else
                dp[i][j] = (pre[j] - pre[j - i]+mod)%mod;
        }
        for(int j=0;j<=20000;j++){
            if(j==0)
                pre[j]=dp[i][j];
            else
                pre[j]=(pre[j-1]+dp[i][j])%mod;
        }
    }
    int t,n,k;
    t=read();
    while(t--){
        n=read();k=read();
        write(dp[n][k]);puts("");
    }
    return 0;
}





 

posted @ 2019-08-29 13:54  eason99  阅读(87)  评论(0编辑  收藏  举报