NOIP2003.加分二叉树

简介: NOIP2003.加分二叉树

题目描述

设一个n个节点的二叉树tree的中序遍历为(l,2,3,…,n),其中数字1,2,3,…,n为节点编号。每个节点都有一个分数(均为正整数),记第j个节点的分数为di,tree及它的每个子树都有一个加分,任一棵子树subtree(也包含tree本身)的加分计算方法如下:

subtree的左子树的加分× subtree的右子树的加分+subtree的根的分数

若某个子树为空,规定其加分为1,叶子的加分就是叶节点本身的分数。不考虑它的空子树。 试求一棵符合中序遍历为(1,2,3,…,n)且加分最高的二叉树tree。

要求输出:

(1)tree的最高加分

(2)tree的前序遍历

输入描述:

第1行:一个整数n(n<30),为节点个数。 第2行:n个用空格隔开的整数,为每个节点的分数(分数<100)。

输出描述:

第1行:一个整数,为最高加分(结果不会超过4,000,000,000)。 第2行:n个用空格隔开的整数,为该树的前序遍历。

示例1

输入

5 5 7 1 2 10

输出

145

3 1 2 4 5

思路

首先我们要有一个思想,对于每一个节点,只有他为根节点时候,他的加分才能最大,所以我们要假设每一个节点为根节点,然后计算他的加分,保留最大值返回。

按照这种思路,我们想到分治做法,构造一个函数求区间[1,n]的最优解

s u b t r e e 的左子树的加分 × s u b t r e e 的右子树的加分+ s u b t r e e 的根的分数 subtree的左子树的加分× subtree的右子树的加分+subtree的根的分数subtree的左子树的加分×subtree的右子树的加分+subtree的根的分数

可以得到

f [ l ] [ r ] = s o l v e ( l , k − 1 ) ∗ s o l v e ( k + 1 , r ) + a [ k ] f[l][r] = solve(l,k-1)*solve(k+1,r) + a[k]f[l][r]=solve(l,k1)solve(k+1,r)+a[k]

其中 k为[l,r]中的一个节点

那么最终答案就是f[1][n]

考虑怎么输出前序遍历的结果呢?我们在记忆化搜索的时候维护一个 root[l][r] = k

代表区间[l,r] 的最优值取的是k节点

那么我们前序遍历的时候只需要遍历 k,[l,k-1],[l,k+1]即可

在递归函数中,如果出现r

代码

N = 32
w = [0] * (N + 1)
f = [[0] * (N + 1) for _ in range(N + 1)]
root = [[0] * (N + 1) for _ in range(N + 1)]
def solve(l, r):
    if f[l][r] != 0:
        return f[l][r]
    if l == r:
        return w[l]
    if l > r:
        return 1
    for i in range(l, r + 1):
        res = solve(l, i - 1) * solve(i + 1, r) + w[i]
        
        if res > f[l][r]:
            f[l][r] = res
            root[l][r] = i
    return f[l][r]
def print_tree(l, r):
    if l == r:
        print(l, end=" ")
        return
    if l > r:
        return
    print(root[l][r], end=" ")
    print_tree(l, root[l][r] - 1)
    print_tree(root[l][r] + 1, r)
n = int(input())
w[1:n+1] = map(int, input().split())
print(solve(1, n))
print_tree(1, n)



目录
相关文章
|
7月前
|
算法
【洛谷 P1003】[NOIP2011 提高组] 铺地毯 题解(数组+贪心算法)
**NOIP2011 提高组问题:铺地毯** 在第一象限的颁奖典礼场地,有$n$张地毯按编号顺序铺设。需找出覆盖特定点$(x, y)$的最上方地毯编号。输入包括地毯坐标和点坐标,输出地毯编号或-1表示未覆盖。 样例:给定3张地毯,点$(2,2)$被第3张地毯覆盖,输出3;另一样例点$(4,5)$未被覆盖,输出-1。 $30\%$数据$n\leq2$,$50\%$数据坐标$\leq100$,$100\%$数据$n\leq10^4$,坐标$\leq10^5$。 解决方法:从下到上遍历地毯,更新覆盖点的地毯编号。 AC代码略。
63 0
|
7月前
【洛谷 P1002】[NOIP2002 普及组] 过河卒 题解(递归+记忆化搜索)
`NOIP2002`普及组的过河卒问题是一个棋盘路径计数挑战。卒从$(0,0)$出发到$(n,m)$,只能向下或向右移动,马在$(c1,c2)$固定,控制某些点。任务是计算不受马阻挡的路径数。输入是目标和马的位置,输出是路径总数。使用动态规划和记忆化搜索避免重复计算,样例输入$(6,6,3,3)$输出$6$。代码中定义了$f(x,y)$计算$(x,y)$处的路径数,利用边界条件和递推关系计算。
83 0
|
7月前
|
存储 C++
【洛谷 P1089】[NOIP2004 提高组] 津津的储蓄计划 题解(循环)
**摘要:** 这是一个关于编程竞赛题目的摘要,题目编号NOIP2004提高组,名为“津津的储蓄计划”。津津每月初从妈妈那里获得300元,需要根据预算决定储蓄。若预计月底有超过或正好100元,她会存储整百金额。如果某月资金不足预算,输出第一个这样的月份加负号;否则,计算年末时津津手中的总金额(储蓄部分加20%)。输入是12个月的预算,输出是一个整数结果。提供的C++代码示例用于处理这个问题,通过迭代计算每个月的资金状况。样例输入和输出展示了不同情况下的结果。
71 0
|
存储 索引
蓝桥杯丨二叉树
蓝桥杯丨二叉树
74 0
|
算法
《蓝桥杯每日一题》二分·AcWing 1460. 我在哪?
《蓝桥杯每日一题》二分·AcWing 1460. 我在哪?
64 0
|
算法 测试技术 Python
第十二届蓝桥杯《杨辉三角》-二分法
第十二届蓝桥杯《杨辉三角》-二分法
103 0
|
测试技术 C语言
题目 2572: 蓝桥杯2020年第十一届省赛真题-子串分值
题目 2572: 蓝桥杯2020年第十一届省赛真题-子串分值
|
存储 移动开发 Python
【蓝桥杯集训·每日一题】AcWing 3555. 二叉树
文章目录 一、题目 1、原题链接 2、题目描述 二、解题报告 1、思路分析 2、时间复杂度 3、代码详解 三、知识风暴 最近公共祖先
91 0