蓝桥杯大赛青少年创意编程C++高级组模拟题-题目的分数值

最近辅导蓝桥杯青少年组,遇到此题网上无题解,自己写了一份。

编程实现: 题目的分数值

程序命名: 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;
}
  • 2
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 5
    评论
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

aq_Seabiscuit

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值