题目描述
观察下面的数字金字塔。
写一个程序来查找从最高点到底部任意处结束的路径,使路径经过数字的和最大。每一步可以走到左下方的点也可以到达右下方的点。
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];
}
}