Educational Codeforces Round 80 C. Two Arrays

http://codeforces.com/contest/1288/problem/C

题意:

用1—n构造两个长为m的数组a,b

满足a[i]<=b[i],a单调不减,b单调不增

求方案数

令dp[i][j][k] 表示构造了长度为i,a[i]=j,b[i]=k的方案数

dp[i][j][k]=Σ dp[i-1][h][p] (h<=p,h<=i,p>=k)

时间复杂度:m*n^4

前缀和、后缀和 优化:

sum[i][j][k] 表示 Σ dp[i][1—j][k—n]

求的时候可以先用后缀和算 sum[i][j][k]=Σ dp[i][j][k—n]

再用前缀和算 sum[i][j][k]=Σ sum[i][1—j][k]

#include<cstdio>

using namespace std;

const int mod=1e9+7;

int dp[11][1001][1001],sum[11][1001][1001];

int main()
{
    int n,m;
    scanf("%d%d",&n,&m);
    for(int i=1;i<=n;++i)
    {
        for(int j=i;j<=n;++j)
        {
            dp[1][i][j]=1;
            sum[1][i][j]=1;
        }
        for(int j=n-1;j>=i;--j) sum[1][i][j]+=sum[1][i][j+1];
    }
    for(int j=1;j<=n;++j)
        for(int k=2;k<=j;++k)
            sum[1][k][j]+=sum[1][k-1][j];
    for(int i=2;i<=m;++i)
    {
        for(int j=1;j<=n;++j)
        {
            for(int k=j;k<=n;++k)
            {
                dp[i][j][k]=sum[i-1][j][k];
                sum[i][j][k]=dp[i][j][k];
            }
            for(int k=n-1;k>=j;--k) sum[i][j][k]=(sum[i][j][k]+sum[i][j][k+1])%mod;
        }
        for(int k=1;k<=n;++k)
            for(int j=2;j<=k;++j)
                sum[i][j][k]=(sum[i][j][k]+sum[i][j-1][k])%mod;
    }
    int ans=0;
    for(int i=1;i<=n;++i)
        for(int j=i;j<=n;++j)
            ans=(ans+dp[m][i][j])%mod;
    printf("%d",ans);
}

原文地址:https://www.cnblogs.com/TheRoadToTheGold/p/12219373.html

时间: 2024-10-09 04:33:42

Educational Codeforces Round 80 C. Two Arrays的相关文章

Educational Codeforces Round 80 C. Two Arrays(组合数快速取模)

You are given two integers nn and mm . Calculate the number of pairs of arrays (a,b)(a,b) such that: the length of both arrays is equal to mm ; each element of each array is an integer between 11 and nn (inclusive); ai≤biai≤bi for any index ii from 1

Educational Codeforces Round 80 (Rated for Div. 2)

\[Educational\ Codeforces\ Round\ 80\ (Rated\ for\ Div.\ 2)\] A.Deadline 打勾函数找最小值,在\(\sqrt{d}\)邻域里找\(x\)最小化\(x+\lceil\frac{d}{x+1}\rceil\)即可 //#pragma comment(linker, "/STACK:1024000000,1024000000") #include<bits/stdc++.h> using namespace

Educational Codeforces Round 80 (Rated for Div. 2)(C - Two Arrays )

C - Two Arrays 题目链接:https://codeforces.com/contest/1288/problem/C 题目: 题意:给你n,m,利用1~n之间的数(可重复)来组成长度为m的数组a,b,要求数组a非递减,数组b非递增,且a数组的数<=b数组中的数,求出a,b数组对数 思路:用动态规划,dp[i][j]是第i个位置放数字j的方案数,根据题意可以将b数组反置然后接在a后面,则该数组长度为2m,为一个非递减序列,则就是求1~n这些数字可重复组成多少种长度为2m的非递减序列,

Educational Codeforces Round 80 (Rated for Div. 2) C - Two Arrays(DP)

???♀? ???♀? ???♀? 题意:从1~n里面选出来m个数字组成a数组,再选出来m个组成b数组,要求a非递减,b非递增,且bi>=ai 1,说是选两个数组其实就是选出来一个长m*2的非递减数组 2,假设要从n的全排列中选出来m长的非递减数组,因为元素是可重复的,最多重复m次,其实就是相当于从下面这个矩阵中选择元素 从这个矩阵中选择元素,每行只能选择一个,枚举我们选出的k个元素的最小值为[ i , j ]位置,那么除去这个元素选择k-1个元素的方案数之和就是k个元素,如图中红色标出位置,最

Educational Codeforces Round 80 A-E简要题解

contest链接:https://codeforces.com/contest/1288 A. Deadline 题意:略 思路:根据题意 x + [d/(x+1)] 需要找到一个x使得上式小于等于n,即x + [d/(x+1) ] <=n,不等式两边同时+1得 x+1 + [d/(x+1)] <=n + 1当且仅当(x+1)2 = d时,式子左边最小,所有只需要判断一下最小值是否<=n+1就可以知道该不等式是否存在x满足题意了,即找到x = √d - 1,判断一下即可. AC代码:

Educational Codeforces Round 80 (Rated for Div. 2)部分题解

A. Deadline 题目链接 题目大意 给你\(n,d\)两个数,问是否存在\(x\)使得\(x+\frac{d}{x+1}\leq n\),其中\(\frac{d}{x+1}\)向上取整. 解题思路 方案一:利用均值不等式公式推导 \(x+\frac{d}{x+1}=x+1+\frac{d}{x+1}-1\geq2\sqrt{d}-1\) 所以 \(\min(x+\frac{x}{d+1})=2\sqrt{d}-1\) 因此去判断\(2\sqrt{d}-1\leq n\)是否成,即\(4\

Educational Codeforces Round 80 (Rated for Div. 2) 题解

Deadline Yet Another Meme Problem Two Arrays Minimax Problem Messenger Simulator Deadline \[ Time Limit: 2 s\quad Memory Limit: 256 MB \] 这是个对勾函数,所以最小的话是在 \(sqrt\) 位置,所以只要找这附近的数字就可以了. view /************************************************************

Educational Codeforces Round 80 (Rated for Div. 2)参加感悟

这次比赛有14000+的人报名,结果我得了266名,创了新纪录. 进过这次比赛,我有回答了1800+. 寒假到了,又可以每次比赛都打了.平时进步很慢,我希望能在寒假有更大的进步. 作为寒假第一场比赛,发挥让我还是很满意的. 开始讲题: A: http://codeforces.com/contest/1288/problem/A 这题太水了,直接是sqrt(d)-1和sqrt(d),如果它们不行,那么其他的也肯定不行. 直接上代码: 1 #include<bits/stdc++.h> 2 #d

Educational Codeforces Round 80 E. Messenger Simulator

http://codeforces.com/contest/1288/problem/E 题意: 有一个长度为n的循环队列,初始第i个位置的数字为i 有m次操作,每次操作把数字x放到队首,原队首与数字x原位置之间的数字在队列中后移一位 输出m次操作过程中,数字i在队列中最靠前和最靠后的位置 若数字i没有移动过, 它最靠前的位置就是初始位置i, 最靠后的位置就是初始位置加上移动过的大于它的数字个数 若数字i被移动过 它最靠前的位置就是1 最靠后的位置是max(初始位置加上第一次移动前移动的大于它的