• Educational Codeforces Round 80 (CF


    传送门:点此跳转至题目

    方法1: dp计数

    状态转移方程:dp( i , j , k ) = sum[ dp( i - 1 , x , y ) ] , 1<=x<=j , k<=y<=n , x<=y;

    dp( i , j , k ) 表示 在长度为 i 的数组,对于数组最后一个数字(即第i个) a[i] = j , b[i] = k 的总情况数;
    其中,a数组是单调不递减的,b数组是单调不递增的,ai<=bi;

    乍一看,这个转移方程的复杂度为 O(m * n ^ 4) ,但其实仔细品味 ,第 i 次的dp(i)值所求的区域和sum 是 第i - 1 次dp(i-1) 的一个矩阵 ,故可用二位前缀和优化,将dp(i) 作为 dp(i-1) 的二维前缀和,复杂度就变成了 O(m * n ^ 2) ;

    dp[i][j][k]=dp[i-1][j][k]+dp[i][j-1][k]+dp[i][j][k+1]-dp[i][j-1][k+1];

    当然还要注意 ai <= bi 的条件:
    在这里插入图片描述
    AC代码:

    #include<cstdio>
    #include<cstring>
    #include<cctype>
    #include<cmath>
    #include<functional>
    #include<algorithm>
    using namespace std;
    typedef long long LL;
    typedef pair<int,int> pii;
    const int N=1e3+5;
    const int inf=0x3f3f3f3f;
    const int mod=1e9+7;
    const double eps=1e-6;
    const long double pi=acos(-1.0L);
    #define ls (i<<1)
    #define rs (i<<1|1)
    #define fi first
    #define se second
    #define pb push_back
    #define mk make_pair
    #define mem(a,b) memset(a,b,sizeof(a))
    LL read()
    {
        LL x=0,t=1;
        char ch;
        while(!isdigit(ch=getchar())) if(ch=='-') t=-1;
        while(isdigit(ch)){ x=10*x+ch-'0'; ch=getchar(); }
        return x*t;
    }
    LL dp[15][N][N];
    int main()
    {
        int n=read(),m=read();
        for(int i=1;i<=n;i++)
            for(int j=n;j>=i;j--)
                dp[1][i][j]=1;
     
        for(int i=2;i<=m;i++)
        {
            for(int j=1;j<=n;j++)
                for(int k=n;k>=j;k--){
                //该转移方程可用二维滚动数组优化空间复杂度
                    dp[i][j][k]=dp[i-1][j][k]+dp[i][j-1][k]+dp[i][j][k+1]-dp[i][j-1][k+1];
                    dp[i][j][k]%=mod;
               
                }
        }
        LL ans=0;
        for(int i=1;i<=n;i++)
            for(int j=n;j>=i;j--){
                ans+=dp[m][i][j];
                ans%=mod;
            }
     
        printf("%lld
    ",ans);
        return 0;
    }
    

    方法二:构造+计数

    将 a 数组 和 b数组的逆序数组连接,构成一个长度为 2 * m 的新数组,保证数组单调不递减,求可构造的总情况数。
    问题转化后,就可以将二维前缀和转化成一维的前缀和了,复杂度O(n * m);
    代码

    L dp[N];
    int main()
    {
        int n=read(),m=read();
        for(int i=1;i<=n;i++) dp[i]=1;
        m<<=1;
        for(int i=2;i<=m;i++)
            for(int j=1;j<=n;j++)
                dp[j]=(dp[j]+dp[j-1])%mod;
        LL ans=0;
        for(int i=1;i<=n;i++) ans=(ans+dp[i])%mod;
        printf("%lld
    ",ans);
        return 0;
    }
    

    方法三:构造+组合数学

    C( 2 * m , n + 2 * m - 1) ;
    从n个数字里选m个数作排序(可选相同的,而能选的相同的数最多2*m个,且这个数在 1-n 种已经出现一次, 故 n + 2 * m -1 个)

  • 相关阅读:
    Spark开发环境搭建(IDEA、Scala、SVN、SBT)
    Spark源码系列:RDD repartition、coalesce 对比
    Scala:类和对象
    申请MS的FastCounter
    code generation part1some OOrelated topic! not completed
    [book]ADO.NET实用指南
    如果FC能把blog的WEB VIEW与AGG VIEW统计起来就方便多了
    Dell 1704
    O'Reilly .NET Framework Essentials, 2nd Edition
    单用户blog系统(一)
  • 原文地址:https://www.cnblogs.com/DeepJay/p/12215029.html
Copyright © 2020-2023  润新知