递归之数字三角形问题

问题描述
在上面的数字三角形中寻找一条从顶部到底边的路径,使得路径上所经过的数字之和最大。路径上的每一步都只能往左下或者右下走。只需要求出这个最大和即可。不必给出具体路径。
三角形的行数大于1小于等于100,数字为0-99
输入格式
5 //三角形行数。下面是三角形
7
3 8
8 1 0
2 7 4 4
4 5 2 6 5
要求输出最大和

递归解决算法思想
用二维数组存放数字三角形
D(r,j):第r行第j个数字(r,j从1开始算)
MaxSum(r,j):从D(r,j)到底边的各条路径中,最佳路径的数字之和。
问题,求MaxSum(1,1)
典型的递归问题
D(r,j)出发,下一步只能走D(r+1,j)或者D(r+1,j+1)。故对于N行的三角形
if(r==N)
MaxSum(r,j)=D(r,j)
else
MaxSum(r,j)=Max{ MaxSum(r+1,j), MaxSum(r+1,j+1)}+D(r,j)

程序代码

#include<iostream>
#include<algorithm>
using namespace std;
#define MAXSIZE 10
int n;
int d[MAXSIZE][MAXSIZE];
int maxSum(int r,int j){
	//当到了最下面一层的时候,就直接等于其值 
	if(r==n){
		return d[r][j];
	}else{
	//到底边的最大路径等于往下走的最大路径和往右下走的最大路径的最大值 
		int x=maxSum(r+1,j);
		int y=maxSum(r+1,j+1);
		return max(x,y)+d[r][j];
	} 
}
int main(){
	cin>>n;
	for(int i=1;i<=n;i++){
		for(int j=1;j<=i;j++){
			cin>>d[i][j];
		}
	}
	cout<<maxSum(2,2);
	return 0;
}

递归解决问题分析
超时?
为什么超时
7(1)
3(1) 8(1)
8(1) 1(2)
2(1) 7(3) 4(3) 4(1)
4(1) 5(4) 2(6) 6(4) 5(1)
回答:重复计算
例如,上面的程序,我们计算maxsum7的时候,会调用maxsum3和8各一次,而第三行的maxsum1会被调用2次,第四行的maxsum7会被调用3次,maxsum3也会被调用3次。把他们全部加起来就是2的n次方。
如果采用递归的办法,深度遍历每条路径,存在大量重复计算。则时间复杂度为2的n次幂,对于n=100,肯定超时
改进
如果每算出一个maxsum(r,j)就保存起来,下次用到其值的时候就直接去用,则可免去重复计算。那么可以用o(n2)时间完成计算。因为三角形的数字总是n(n+1)/2。
递归解决问题改进程序代码

#include<iostream>
#include<algorithm>
using namespace std;
#define MAXSIZE 10
int n;
int d[MAXSIZE][MAXSIZE];
int maxSum[MAXSIZE][MAXSIZE];		//存储从某一点到最底边的最大路径值 
int MaxSum(int r,int j){
	if(maxSum[r][j]!=-1){
		return maxSum[r][j]; 
	}
	//当到了最下面一层的时候,就直接等于其值 
	if(r==n){
		maxSum[r][j]=d[r][j];
	}else{
	//到底边的最大路径等于往下走的最大路径和往右下走的最大路径的最大值 
		int x=MaxSum(r+1,j);
		int y=MaxSum(r+1,j+1);
		maxSum[r][j]=max(x,y)+d[r][j];
	} 
	return maxSum[r][j];
}
int main(){
	cin>>n;
	for(int i=1;i<=n;i++){
		for(int j=1;j<=i;j++){
			cin>>d[i][j];
			//初始化maxsum数组,当未被计算时,其值为-1 
			maxSum[i][j]=-1;	
		}
	}
	cout<<MaxSum(2,2);
	return 0;
}

  • 2
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值