【蓝桥杯】算法提高 最大值路径(递归+动态规划)(C++)

ADV-293 最大值路径

  • 问题描述
  • 思路分析
  • 代码实现

问题描述

题目链接:最大值路径
资源限制
时间限制:1.0s 内存限制:256.0MB
问题描述
  刷微博,编程序。如下图所示,@北京发布 提出了如下“头脑震荡”问题。对此问题做一般化描述:
  有n阶方阵,从矩阵的左下角元素为起点,从行或列(水平或垂直)两个方向上移动,直到右上角。求出有多少条路径可以使得经过的元素累加值最大,最大值是多少。
【蓝桥杯】算法提高 最大值路径(递归+动态规划)(C++)_第1张图片
输入格式
  共有n+1行。
  第一行整数n,表示矩阵的阶数,2<=n<=10。
  第二行起,每行n个整数,以空格分隔,共n行。。
输出格式
  一行,两个空格分隔的数,第一个表示最大值路径的条数,第二个表示最大值。

样例输入
5
4 5 4 5 6
2 6 5 4 6
2 6 6 5 2
4 5 2 2 5
5 2 5 6 4

样例输出
3 47

思路分析

思路:题目中要求的是从矩阵的左下角元素为起点,从行或列(水平或垂直)两个方向上移动,直到右上角。问有多少条路径可以使经过的元素累加值最大,且要求出最大值。
1.考虑如何求得最大值?如下所示,手动计算从左下角到右上角经过元素的累加值,可以看出,每经过一个点,都会比较当前点左边和下面比较大的那个值,再加上其本身得到当前点的值,且走到右上角时元素值是最大的。
因此考虑用动态规划来进行从左下角到右上角元素值累加的实现: (1)从左下角点开始,起点等于其本身;
(2)对于第n行元素,则任一右边的数等于其左边数加上其本身 ; (3)对于第1列元素,则任一上边的数等于其下边数加上其本身 ;
(4)其余情况,每次比较要求的当前位置累加值的下面和其左面值的大小,得到大的值再加上其本身。

2.考虑找到最大元素的路径条数。在已求出元素累加值得情况下,可以逆向对矩阵进行递归。每次到一个点,比较其左边和下边哪个值大(大的值说明从左上角开始经过的元素累加值更大),对大的那个值继续进行递归,如果两个值相等,则说明有两条路径能到达元素值累加值最大的点(即右上角),因此路径数要加1。

[17 31 35 40 47
 13 26 31 35 41
 11 20 26 31 33
 9  14 16 20 27
 5  7  12 18 22]

代码实现

#include 
using namespace std;
#define MAXSIZE 15
int n;
int cnt=1;
int a[MAXSIZE][MAXSIZE];
int f[MAXSIZE][MAXSIZE];


int dfs(int x,int y)//从右上角开始搜索 
{
	if(x>=n&&y<=1)return 0;//找到左下角或出界,退出 
	if(f[x+1][y]>f[x][y-1])dfs(x+1,y);//如果下边的值大于左边的值 
	if(f[x][y-1]>f[x+1][y])dfs(x,y-1);//如果左边的值大于下边的值 
	if(f[x][y-1]==f[x+1][y])//如果两点的路径之和相等 
	{
		cnt++;//路径数加1 
		dfs(x,y-1);
		dfs(x+1,y);
	} 
	return cnt;
}

int main()
{
	cin>>n;
	for(int i=1;i<=n;i++)
	{
		for(int j=1;j<=n;j++)
		{
			cin>>a[i][j];
		}
	}
	for(int i=n;i>0;i--)
	{
		for(int j=1;j<=n;j++)
		{
			if(i==n&&j==1)f[i][j]=a[i][j];//从左下角开始,起点等于它本身 
			else if(i==n)//如果在第n行,则任一右边的数等于其左边数加上其本身 
				f[i][j]=f[i][j-1]+a[i][j];
			else if(j==1)//如果在第1列,则任一上边的数等于其下边数加上其本身 
				f[i][j]=f[i+1][j]+a[i][j];
			else//其他情况,在两种情况中选择大值加上它本身 
				f[i][j]=max(f[i][j-1],f[i+1][j])+a[i][j];
		}
	} 
	cout<<dfs(1,n)<<" "<<f[1][n]<<endl;
	return 0;
}

【蓝桥杯】算法提高 最大值路径(递归+动态规划)(C++)_第2张图片

你可能感兴趣的:(蓝桥杯)