洛谷---P1216 [USACO1.5][IOI1994]数字三角形 Number Triangles

题目描述

观察下面的数字金字塔。

写一个程序来查找从最高点到底部任意处结束的路径,使路径经过数字的和最大。每一步可以走到左下方的点也可以到达右下方的点。

        7 
      3   8 
    8   1   0 
  2   7   4   4 
4   5   2   6   5 

在上面的样例中,从 7→3→8→7→5 的路径产生了最大

输入格式

第一个行一个正整数 rr ,表示行的数目。

后面每行为这个数字金字塔特定行包含的整数。

输出格式

单独的一行,包含那个可能得到的最大的和。

输入输出样例

输入

5
7
3 8
8 1 0
2 7 4 4
4 5 2 6 5 

输出

30

说明/提示

【数据范围】
对于 100% 的数据,1≤r≤1000,所有输入在[0,100] 范围内。

题目翻译来自NOCOW。

USACO Training Section 1.5

IOI1994 Day1T1

原题链接:[USACO1.5][IOI1994]数字三角形 Number Triangles - 洛谷

 

方法一:dfs深搜所有可能 最后找到最大的,但递归树过于庞大必须剪枝 采用记忆化dfs

 

 

import java.util.Arrays;
import java.util.Scanner;
public class Main {
	static int[][] arr;
	static int[][] pd;
	public static void main(String[] args) {
		Scanner sc = new Scanner(System.in);
		int r = sc.nextInt(),c=0;
		arr = new int[r][];
		pd = new int[r][];
		for(int i =0;i<r;i++) {
			c++;
			arr[i] = new int[c];
			pd[i] = new int[c];
			for(int j =0;j<c;j++) {
				arr[i][j] = sc.nextInt();
				pd[i][j] = -1;
			}
		}
		int res = dfs(r,0,0);
		System.out.println(res);
		sc.close();
	}
	public static int dfs(int n,int k,int y) {
		if(k == n-1) {
			return arr[k][y];
		}
		if(pd[k][y] == -1) {
			pd[k][y] = arr[k][y]+Math.max(dfs(n,k+1,y),dfs(n,k+1,y+1));
		}
		return pd[k][y];
	}
}

方法二:动态规划 从三角形最下层往上递推 每次都选下一层左边和后边最大的

import java.util.Arrays;
import java.util.Scanner;
public class Main {
	static int[][] arr;
	static int[][] pd;
	public static void main(String[] args) {
		Scanner sc = new Scanner(System.in);
		int r = sc.nextInt(),c=0;
		arr = new int[r][];
		pd = new int[r][];
		for(int i =0;i<r;i++) {
			c++;
			arr[i] = new int[c];
			pd[i] = new int[c];
			for(int j =0;j<c;j++) {
				arr[i][j] = sc.nextInt();
				pd[i][j] = -1;
			}
		}
		dy(r);
		sc.close();
	}
	public static void dy(int r) {
		int[] dp = new int[r];
		for(int i =0;i<r;i++) {
			dp[i] = arr[r-1][i];
		}
		for(int i = r-2;i>=0;i--) {
			for(int j =0;j<arr[i].length;j++) {
				dp[j] = arr[i][j] + Math.max(dp[j], dp[j+1]);
			}
		}
		System.out.println(dp[0]);
	}
	public static int dfs(int n,int k,int y) {
		if(k == n-1) {
			return arr[k][y];
		}
		if(pd[k][y] == -1) {
			pd[k][y] = arr[k][y]+Math.max(dfs(n,k+1,y),dfs(n,k+1,y+1));
		}
		return pd[k][y];
	}
}

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值