解题思路:




注意事项:

为什么是四维?


因为里面有坐标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.0分

1 人评分

C语言网提供由在职研发工程师或ACM蓝桥杯竞赛优秀选手录制的视频教程,并配有习题和答疑,点击了解:

一点编程也不会写的:零基础C语言学练课程

解决困扰你多年的C语言疑难杂症特性的C语言进阶课程

从零到写出一个爬虫的Python编程课程

只会语法写不出代码?手把手带你写100个编程真题的编程百练课程

信息学奥赛或C++选手的 必学C++课程

蓝桥杯ACM、信息学奥赛的必学课程:算法竞赛课入门课程

手把手讲解近五年真题的蓝桥杯辅导课程

评论列表 共有 0 条评论

暂无评论