本文介绍: 【蓝桥杯冲冲冲】动态规划学习 [NOIP2003 提高组] 加分二叉树

【蓝桥杯冲冲冲】动态规划学习 [NOIP2003 提高组] 加分二叉树

蓝桥杯备赛 | 洛谷做题打卡day24

  • [NOIP2003 提高组] 加分二叉树

    题目描述

    设一个

    n

    n

    n 个节点的二叉树

    tree

    text{tree}

    tree 的中序遍历为

    (

    1

    ,

    2

    ,

    3

    ,

    ,

    n

    )

    (1,2,3,ldots,n)

    (1,2,3,,n),其中数字

    1

    ,

    2

    ,

    3

    ,

    ,

    n

    1,2,3,ldots,n

    1,2,3,,n 为节点编号。每个节点都有一个分数(均为正整数),记第

    i

    i

    i 个节点的分数为

    d

    i

    d_i

    di

    tree

    text{tree}

    tree 及它的每个子树都有一个加分,任一棵子树

    subtree

    text{subtree}

    subtree(也包含

    tree

    text{tree}

    tree 本身)的加分计算方法如下:

    subtree

    text{subtree}

    subtree 的左子树的加分

    ×

    times

    ×

    subtree

    text{subtree}

    subtree 的右子树的加分

    +

    +

    +

    subtree

    text{subtree}

    subtree 的根的分数。

    若某个子树为空,规定其加分为

    1

    1

    1,叶子的加分就是叶节点本身的分数。不考虑它的空子树。

    试求一棵符合中序遍历为

    (

    1

    ,

    2

    ,

    3

    ,

    ,

    n

    )

    (1,2,3,ldots,n)

    (1,2,3,,n) 且加分最高的二叉树

    tree

    text{tree}

    tree。要求输出

    1. tree

      text{tree}

      tree 的最高加分。

    2. tree

      text{tree}

      tree 的前序遍历。

    输入格式

    1

    1

    1

    1

    1

    1 个整数

    n

    n

    n,为节点个数。

    2

    2

    2

    n

    n

    n 个用空格隔开的整数,为每个节点的分数

    输出格式

    1

    1

    1

    1

    1

    1 个整数,为最高加分($ Ans le 4,000,000,000$)。

    2

    2

    2

    n

    n

    n 个用空格隔开的整数,为该树的前序遍历。

    样例 #1

    样例输入 #1

    5
    5 7 1 2 10
    

    样例输出 #1

    145
    3 1 2 4 5
    

    提示

    数据规模与约定

    对于全部的测试点,保证

    1

    n

    <

    30

    1 leq n< 30

    1n<30,节点的分数是小于

    100

    100

    100 的正整数,答案不超过

    4

    ×

    1

    0

    9

    4 times 10^9

    4×109

    在这里插入图片描述

思路

一道入门的区间dp,当然,根据写法不同你还可以把它归类为树形dp或者记忆化搜索,其实都无所谓啦。
作为一道入门题,我们完全可以“显然”地做出来,但是在这里还是想和大家回顾下动态规划以及区间动规。

Q:dp特点是什么?
A:dp把原问题视作若干个重叠的子问题的逐层递进,每个子问题的求解过程都会构成一个“阶段”,在完成一个阶段后,才会执行下一个阶段。
Q:dp要满足无后效性,什么叫无后效性?
A:已经求解的子问题不受后续阶段的影响。

有人觉得dp很抽象,那是因为没有一步一步来想,直接听别人的结论,我们在这里以这道题为例,一步一步来推导。

首先,我们要做的就是设计状态,其实就是设计dp数组的含义,它要满足无后效性。
关注这个 左子树*右子树+根 我只要知道左子树分数和右子树分数和根的分数(已给出),不就可以了吗?管他子树长什么样!
所以,我们f数组存的就是最大分数,怎么存呢?
我们发现:子树是一个或多个节点的集合。

题解代码

学会利用新知,自己多试试并尝试积攒一些固定解答方案,debug,以下是题解代码 ~

#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
const int MAXN = 50;
typedef long long ll;
ll n;
ll f[MAXN][MAXN], root[MAXN][MAXN];

void print(ll l, ll r) {
	if (l > r)return;
	printf("%lld ", root[l][r]);
	if (l == r)return;
	print(l, root[l][r] - 1);
	print(root[l][r]+1,r);
}

int main() {
	scanf("%lld", &n);
	for (int i = 1; i <= n; i++)scanf("%lld", &f[i][i]),f[i][i-1]=1, root[i][i] = i;
	for (int len = 1; len < n; ++len) {
		for (int i = 1; i + len <= n; ++i) {
			int j = i + len;
			f[i][j] = f[i + 1][j] + f[i][i];//默认它的左子树为空,如果有的话,这肯定不是最优解
			root[i][j] = i;//默认从起点选根
			for (int k = i + 1; k < j; ++k) {
				if (f[i][j] < f[i][k - 1] * f[k + 1][j] + f[k][k]) {
					f[i][j] = f[i][k - 1] * f[k + 1][j] + f[k][k];
					root[i][j] = k;
				}
			}
		}
	}
	cout << f[1][n] << endl;
	print(1, n);
	return 0;
}

我的一些话

  • 今天学习动态规划,dp属于比较难的部分,需要多动脑,多思考思路还是很好掌握的,虽然一次性AC有一定难度,需要通盘的考虑和理解,以及扎实的数据结构基础才能独立写出AC代码。但无论难易,大家都要持续做题,保持题感喔!一起坚持(o´ω`o)

  • 如果有非计算机专业的uu自学的话,关于数据结构的网课推荐看b站上青岛大学王卓老师的课,讲的很细致,有不懂都可以私信我喔

  • 总结来说思路很重要,多想想,多在草稿纸上画画,用测试数据多调试,debug后成功编译并运行出正确结果真的会感到很幸福!

  • 关于之前蓝桥杯备赛的路线和基本方法、要掌握的知识,之前的博文我都有写,欢迎大家关注我,翻阅自取哦~

  • 不管什么都要坚持吧,三天打鱼两天晒网无法形成肌肉记忆和做题思维,该思考的时候一定不要懈怠,今天就说这么多啦,欢迎评论留言,一起成长:)

原文地址:https://blog.csdn.net/m0_73246124/article/details/135971034

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。

如若转载,请注明出处:http://www.7code.cn/show_65011.html

如若内容造成侵权/违法违规/事实不符,请联系代码007邮箱:suwngjj01@126.com进行投诉反馈,一经查实,立即删除!

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注