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

文章讲述了在解决数字三角形问题中,如何从深度优先搜索(DFS)优化为记忆化搜索,最后通过动态规划求得最优路径,以提高C++程序的运行效率并成功AC。
摘要由CSDN通过智能技术生成

题目链接:P1216 [USACO1.5] [IOI1994]数字三角形 Number Triangles - 洛谷 | 计算机科学教育新生态 (luogu.com.cn)

解题思路:

最优路径问题,首先想到dfs深度优先搜索,一直往下走再回溯上一格换个方向走

下面是c++代码:

#include<iostream>
#include<algorithm>
using namespace std;
int arr[1010][1010];
int n;
int dfs(int x, int y) {
	if (x > n || y > n)return 0;
	else return max(dfs(x + 1, y), dfs(x + 1, y + 1)) + arr[x][y];
}
int main()
{
	cin >> n;
	for (int i = 1; i <= n; i++) {
		for (int j = 1; j <= i; j++) {
			cin >> arr[i][j];
		}
	}
	cout << dfs(1, 1);
	return 0;
}

但是这样超时了,可以优化一下,想到记忆化搜索,就是把经过的最大路径存起来,遇到直接使用即可

c++代码:

#include<iostream>
#include<algorithm>
using namespace std;
int arr[1010][1010];
int value[10001][10001];
int n;
int dfs(int x, int y) {
	if (value[x][y])return value[x][y];
	int sum = 0;
	if (x > n || y > n)return 0;
	else sum = max(dfs(x + 1, y), dfs(x + 1, y + 1)) + arr[x][y];
	value[x][y] = sum;
	return sum;
}
int main()
{
	cin >> n;
	for (int i = 1; i <= n; i++) {
		for (int j = 1; j <= i; j++) {
			cin >> arr[i][j];
		}
	}
	cout << dfs(1,1);
	return 0;
}

好多了,但是还是有一个测试点t了,直接转换成动态规划

c++代码:

#include<iostream>
#include<algorithm>
using namespace std;
int arr[1010][1010];
int value[10001][10001];
int n;
int dfs(int x, int y) {
	if (value[x][y])return value[x][y];
	int sum = 0;
	if (x > n || y > n)return 0;
	else sum = max(dfs(x + 1, y), dfs(x + 1, y + 1)) + arr[x][y];
	value[x][y] = sum;
	return sum;
}
int main()
{
	cin >> n;
	for (int i = 1; i <= n; i++) {
		for (int j = 1; j <= i; j++) {
			cin >> arr[i][j];
		}
	}
	for (int i = n; i >= 1; i--) {
		for (int j = 1; j <= n; j++) {
			value[i][j] = max(value[i + 1][j], value[i + 1][j + 1]) + arr[i][j];
		}
	}
	cout << value[1][1];
	return 0;
}

ac了,遇到类似的题都可以先dfs再剪枝再转换成动态规划,可以直接套用模板

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值