赞
踩
最近辅导蓝桥杯青少年组,遇到此题网上无题解,自己写了一份。
编程实现: 题目的分数值
程序命名: score.cpp
题目描述:
有n个问题,现在请你给这n个问题分配分值。
n个问题已经按从简单到困难排好序,第i个问题的分值是Ai,n个问题的分值满足如下关系:1<=A1<=A2<=…<=An<=n。不同的问题可以具有相同的分值。
主办方希望:解决更多问题的参赛者的排名更高。因此,对于任何解决了k(1 <=k<=n-1)个问题的参赛者,其分数总和一定要小于解决了任何k+1个问题的参赛者的分数总和。
你有几种分配分值的方法?将答案对素数m取余后输出。
输入:
整数n和m
其中2<=n<=5000,9x108 <m<109,m为素数。
输出:
分值分配的方案数对m取余后的数字
样例输入1:
2 998244353
样例输出1:
3
样例1说明:
2个题的分值分配有3种方案:(1,1),(1,2),(2,2)。
样例输入2:
3 998244353
样例输出2:
7
样例2说明:
3个题的分值分配有7种方案:(1,1,1),(1,2,2),(1,3,3),(2,2,2),(2,2,3),(2,3,3),(3,3,3)
首先对题目分析。
要保证做K+1个题目得分比做K个高,则要求满足任意两个题目分值大于任意第三个题目,又A1和A2为分值最小的两个题目。则有:Ai>A1+A2。
A1和A2的分值可以通过枚举选取,从第三个题目开始,每个题目的分值须满足Ai>=Ai-1,且Ai<=min(n,A1+A2-1)。
然后递推,定义子问题。
后续的分值选取方案数与当前分值选取有关。
设dp[i][k]表示后续还剩i个未确定分值的题目,且当前题目可选方案为k种的总方案数。每个题目的分值方案数满足递推式:
递推边界为:dp[1][k] = k
观察递推式具有累加性质,可优化为:
对定义的dp值先做预处理,原问题等价于先取了前两个数,后续的A3按照约束有k1种取法,k1 = min(A1,n-A2+1),依次是A2 ~ k1,按加法原理,此时对答案贡献dp[n-2][k1]
由数论基本定理,对答案取模等价于对加法运算中各步骤结果取模的总和再取模。
#include<iostream> using namespace std; const int MAXN = 5005; int dp[MAXN][MAXN]; int main() { int n,m; cin>>n>>m; for(int k=1;k<=n;k++) dp[1][k] = k; for(int i=2;i<=n;i++) // 后面还有几个数 { for(int k=1;k<=n;k++) //当前阶段有多少种选择 { if(k==1) dp[i][k] = dp[i-1][k]; else dp[i][k] = (dp[i][k-1] + dp[i-1][k]) % m; } } int ans = 0; for(int i=1;i<=n;i++) { for(int j=i;j<=n;j++) { int k1= min(i,n-j+1); ans = (ans + dp[n-2][k1]) % m; } } if(n==2) ans = 3; cout<<ans<<endl; return 0; }
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。