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月前
【洛谷 P1980】[NOIP2013 普及组] 计数问题 题解(取余)
NOIP2013普及组计数问题,求区间[1, n]内数字x出现的次数。输入为n和x,输出x的出现次数。样例输入11 1,输出4。代码通过逐位检查每个数是否等于x来计数,适用于$n\leq10^6$,$0\leq x\leq 9$的情况。
82 0
|
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代码略。
64 0
|
7月前
【洛谷 P1002】[NOIP2002 普及组] 过河卒 题解(递归+记忆化搜索)
`NOIP2002`普及组的过河卒问题是一个棋盘路径计数挑战。卒从$(0,0)$出发到$(n,m)$,只能向下或向右移动,马在$(c1,c2)$固定,控制某些点。任务是计算不受马阻挡的路径数。输入是目标和马的位置,输出是路径总数。使用动态规划和记忆化搜索避免重复计算,样例输入$(6,6,3,3)$输出$6$。代码中定义了$f(x,y)$计算$(x,y)$处的路径数,利用边界条件和递推关系计算。
84 0
|
7月前
【洛谷 P1093】[NOIP2007 普及组] 奖学金 题解(结构体排序)
**NOIP2007普及组奖学金问题**:根据学生语文、数学、英语三科成绩计算总分并排序。若总分相同,按语文成绩高者优先,再相同则学号小者靠前。程序需输出前5名学生的学号和总分。输入包括学生人数`n`和每人的三科成绩,输出为5行结果。示例输入和输出已给出,代码通过定义结构体和自定义比较器实现排序。
55 0
|
7月前
|
存储 C++
【洛谷 P1089】[NOIP2004 提高组] 津津的储蓄计划 题解(循环)
**摘要:** 这是一个关于编程竞赛题目的摘要,题目编号NOIP2004提高组,名为“津津的储蓄计划”。津津每月初从妈妈那里获得300元,需要根据预算决定储蓄。若预计月底有超过或正好100元,她会存储整百金额。如果某月资金不足预算,输出第一个这样的月份加负号;否则,计算年末时津津手中的总金额(储蓄部分加20%)。输入是12个月的预算,输出是一个整数结果。提供的C++代码示例用于处理这个问题,通过迭代计算每个月的资金状况。样例输入和输出展示了不同情况下的结果。
71 0
|
7月前
|
C++
【洛谷 P1047】[NOIP2005 普及组] 校门外的树 题解(位集合)
**NOIP2005普及组问题:**给定长度为$l$的马路,上面等距种植着树,需移除位于建造地铁区域的树。输入包含马路长度和区域数,以及各区域起止点,输出移树后剩余树的数量。样例输入:$l=500$, $m=3$,输出:$298$。$20\%$数据无区域重合,$1 \leq l \leq 10^4$,$1 \leq m \leq 100$。解决方案利用位集合(bitset)表示树的状态,遍历区域将树设为0,最后统计1的数量。AC代码使用C++实现。
35 0
[算法刷题题解笔记] 洛谷 P1011 [NOIP1998 提高组] 车站 [数学|斐波那契|推导]
[算法刷题题解笔记] 洛谷 P1011 [NOIP1998 提高组] 车站 [数学|斐波那契|推导]
|
8月前
|
测试技术
题目1444:蓝桥杯2014年第五届真题斐波那契
题目1444:蓝桥杯2014年第五届真题斐波那契
60 0
|
机器学习/深度学习 算法
蓝桥杯丨回溯法
蓝桥杯丨回溯法
56 0
|
算法
《蓝桥杯每日一题》二分·AcWing 1460. 我在哪?
《蓝桥杯每日一题》二分·AcWing 1460. 我在哪?
64 0