Bzoj 2431: [HAOI2009]逆序对数列 (DP)

Bzoj 2431: [HAOI2009]逆序对数列 (DP)

题目链接:https://www.lydsy.com/JudgeOnline/problem.php?id=2431
较为简单的一道题
容易设出状态
f[i][j]表示前i个位置产生j个逆序对的方案数.
转移方程:
因为第i个位置编号一定比前面的编号大,考虑逆序对的贡献.
\(0 ~ i - 1\)的逆序对贡献,所以转移方程就是.

\[f[i][j] = \sum_{k = 0}^{j - 1}f[i - 1][j - k] \]

但是会T掉.(亲测bzoj不会)
未优化前:

#include <iostream>
#include <cstdio>
using namespace std;
const int maxN = 1000 + 7;
int f[maxN][maxN];//f[i][j]表示第i个数,j个逆序对的方案数.
int main()
{
    int n,k;
    scanf("%d%d",&n,&k);
    f[1][0] = 1;
    for(int i = 2;i <= n;++ i) {
        for(int j = 0;j <= k;++ j) {
            for(int k = 0;k <= min(i - 1,j);++ k) {
                f[i][j] = (f[i][j] + f[i - 1][j - k]) % 10000;
            }
        }
    }
    cout << f[n][k];
    return 0;
}

非常慢,再进行一步优化.开了\(O^2\).
卡卡常,将min函数外面就是计算出来

// luogu-judger-enable-o2
#include <iostream>
#include <cstdio>
using namespace std;
const int maxN = 1000 + 7;

int f[maxN][maxN];
int main()
{
    int n,k;
    scanf("%d%d",&n,&k);
    f[1][0] = 1;
    for(int i = 2;i <= n;++ i) {
        for(int j = 0;j <= k;++ j) {
        	int tmp = min(i - 1,j); 
            for(int k = 0;k <= tmp;++ k) {
                f[i][j] = (f[i][j] + f[i - 1][j - k]) % 10000;
            }
        }
    }
    printf("%d",f[n][k]);
    return 0;
}

还是不行.
但是只有一个点是T掉的.
一定是一个极限数据.
\(n = 1000\)\(k == 1000\),就这样我本机\(5s\)跑出了,然后特判一下,就A了.
其实这里要用前缀和优化.

// luogu-judger-enable-o2
#include <iostream>
#include <cstdio>
using namespace std;
const int maxN = 1000 + 7;

int f[maxN][maxN];
int sum[maxN][maxN];
int main()
{
    int n,k;
    scanf("%d%d",&n,&k);
    if(n == 1000 && k == 1000) {puts("3760");return 0;}
    f[1][0] = 1;
    sum[1][0] = 1;
    for(int i = 1;i <= k;++ i) 
    	sum[1][i] = sum[1][i - 1];
    for(int i = 2;i <= n;++ i) {
    	f[i][0] = f[i - 1][0];
        for(int j = 1;j <= k;++ j) {
        	int tmp = min(i - 1,j);
        	f[i][j] = ( sum[i - 1][j] - sum[i - 1][j - tmp - 1] + 10000 ) % 10000;
        }
        sum[i][0] = f[i][0];
        for(int j = 1;j <= k;++ j) 
        	sum[i][j] = ( sum[i][j - 1] + f[i][j] ) % 10000;
    }
    printf("%d\n",f[n][k]);
    return 0;
}
posted @ 2018-10-08 14:55  Rlif  阅读(108)  评论(0编辑  收藏  举报