解题思路:
注意事项:
为什么是四维?
因为里面有坐标i,j从两个方向来,还有k件物品需要维护,另外还有最后一件物品的价值c需要维护,因为需要c来比较,是否取不取,这样就可以表示出,这个问题了。
参考代码:
#include <stdio.h>
int dp[51][51][15][15];//dp[i][j][k][v]表示走到坐标为[i,j],取了k个数,最后一个物品的价值是v的方案数
int c[51][51]; //输入得n*m矩阵
int main()
{
int i,j,u,l;
int n,m,k;
scanf("%d%d%d",&n,&m,&k); //输入
for(i=1;i<=n;i++)
{
for(j=1;j<=m;j++)
{
scanf("%d",&c[i][j]);
}
}
dp[1][1][0][-1]=1; //不取左上角 边界赋值 避免无法取到价值为0的物品 将最后一个参数置为-1
dp[1][1][1][c[1][1]]=1;//取左上角
for(i=1;i<=n;i++) //遍历矩阵
{
for(j=1;j<=m;j++)
{
if(i==1&&j==1)
{
continue;
}
for(u=0;u<=k;u++) //枚举所有方案数 选了u个物品
{
for(l=-1;l<=12;l++) //枚举最后一个物品的价值(最大价值)
{
//不取(i,j)位置上的物品
int *num=&dp[i][j][u][l];
(*num)=((*num)+dp[i-1][j][u][l])%1000000007; //从上方下来
(*num)=((*num)+dp[i][j-1][u][l])%1000000007; //从左边过来
//取(i,j)位置上的物品
if(u>0&&c[i][j]==l) //枚举取当前[i,j]位置上的物品 当前物品的价值必须等于l
{
for(int w=-1;w<l;w++) //枚举上一个方案中小于最后一个物品价值的所有可能
{
(*num)=((*num)+dp[i-1][j][u-1][w])%1000000007;
(*num)=((*num)+dp[i][j-1][u-1][w])%1000000007;
}
}
}
}
}
}
int num=0;
for(i=-1;i<=12;i++)
{
num=(num+dp[n][m][k][i])%1000000007;
}
printf("%d",num);
return 0;
}
0.0分
1 人评分
C语言网提供由在职研发工程师或ACM蓝桥杯竞赛优秀选手录制的视频教程,并配有习题和答疑,点击了解:
一点编程也不会写的:零基础C语言学练课程
解决困扰你多年的C语言疑难杂症特性的C语言进阶课程
从零到写出一个爬虫的Python编程课程
只会语法写不出代码?手把手带你写100个编程真题的编程百练课程
信息学奥赛或C++选手的 必学C++课程
蓝桥杯ACM、信息学奥赛的必学课程:算法竞赛课入门课程
手把手讲解近五年真题的蓝桥杯辅导课程
发表评论 取消回复