http://www.luogu.org/problem/show?pid=1040

题目描述

设一个n个节点的二叉树tree的中序遍历为(1,2,3,…,n),其中数字1,2,3,…,n为节点编号。每个节点都有一个分数(均为正整数),记第i个节点的分数为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个用空格隔开的整数,为该树的前序遍历。

一开始写的是在算可以取得的最大值的时候计算每个节点的左右儿子,然而这个思路并不对...因为数据规模较小,DP求完最大值后可以再记忆化搜索一次来打印先序遍历,因为第一次已经求出了所有情况的值,所以第二次DP很快

 #include<iostream>
 #include<cstdio>
 #include<cstring>
 using namespace std;
 struct Node{
     int root;
     long long val;
 };
 ;
 long long mem[maxn][maxn];
 int v[maxn][maxn],lson[maxn],rson[maxn],num[maxn],fa[maxn][maxn],n;
 int init(){
     memset(v,,sizeof(v));
     memset(lson,,sizeof(lson));
     memset(rson,,sizeof(rson));
     memset(fa,,sizeof(fa));
 }
 Node dp(int l,int r){
     if(v[l][r]) return (Node){fa[l][r],mem[l][r]};
     v[l][r]=;
     ,tfa=-;
     if(l==r){
         mem[l][r]=num[l];
         fa[l][r]=l;
         return (Node){l,num[l]};
     }
     tfa=l;ans=dp(l+,r).val+num[l];
     ).val+num[r];
     if(x>ans) ans=x,tfa=r;
     ;i<r;i++){
         x=dp(l,i-).val*dp(i+,r).val+num[i];
         if(x>ans) ans=x,tfa=i;
     }
     ,r).root;
     ).root;
     else{
         lson[tfa]=dp(l,tfa-).root;
         rson[tfa]=dp(tfa+,r).root;
     }
     mem[l][r]=ans,fa[l][r]=tfa;
     return (Node){tfa,ans};
 }
 void print(int l,int r){
     if(l==r){
         printf("%d ",l);
         return;
     }
     ,r).val+num[l];int f=l;
     ).val+num[r];
     if(x>ans) ans=x,f=r;
     ;i<r;i++){
         x=dp(l,i-).val*dp(i+,r).val+num[i];
         if(x>ans) ans=x,f=i;
     }
     printf("%d ",f);
     if(f==l){
         print(l+,r);
     }else if(f==r){
         print(l,r-);
     }else{
         print(l,f-);
         print(f+,r);
     }
 }
 void dfs(int u){
     printf("%d ",u);
     if(lson[u]) dfs(lson[u]);
     if(rson[u]) dfs(rson[u]);
 }
 int main()
 {
     scanf("%d",&n);
     init();
     ;i<=n;i++) scanf("%d",&num[i]);
     printf(,n).val);
     //dfs(dp(1,n).root);
     print(,n);
     ;
 }

NOIP2003 加分二叉树的更多相关文章

  1. cogs 106. [NOIP2003] 加分二叉树(区间DP)

    106. [NOIP2003] 加分二叉树 ★☆   输入文件:jfecs.in   输出文件:jfecs.out   简单对比时间限制:1 s   内存限制:128 MB [问题描述] 设 一个 n ...

  2. NOIP2003加分二叉树[树 区间DP]

    题目描述 设一个n个节点的二叉树tree的中序遍历为(1,2,3,…,n),其中数字1,2,3,…,n为节点编号.每个节点都有一个分数(均为正整数),记第i个节点的分数为di,tree及它的每个子树都 ...

  3. 【题解】NOI2009二叉查找树 + NOIP2003加分二叉树

    自己的思维能力果然还是太不够……想到了这棵树所有的性质即中序遍历不变,却并没有想到怎样利用这一点.在想这道题的过程中走入了诸多的误区,在这里想记录一下 & 从中吸取到的教训(原该可以避免的吧) ...

  4. NOIP-2003 加分二叉树

    题目描述 设一个n个节点的二叉树tree的中序遍历为(1,2,3,…,n),其中数字1,2,3,…,n为节点编号.每个节点都有一个分数(均为正整数),记第i个节点的分数为di,tree及它的每个子树都 ...

  5. NOIP2003加分二叉树

    题目描述 设一个n个节点的二叉树tree的中序遍历为(1,2,3,…,n),其中数字1,2,3,…,n为节点编号.每个节点都有一个分数(均为正整数),记第ii个节点的分数为di,treedi,tree ...

  6. CJOJ 1010【NOIP2003】加分二叉树 / Luogu 1040 加分二叉树(树型动态规划)

    CJOJ 1010[NOIP2003]加分二叉树 / Luogu 1040 加分二叉树(树型动态规划) Description 设 一个 n 个节点的二叉树 tree 的中序遍历为( 1,2,3,-, ...

  7. 加分二叉树 vijos1991 NOIP2003第三题 区间DP/树形DP/记忆化搜索

    描述 设一个n个节点的二叉树tree的中序遍历为(l,2,3,-,n),其中数字1,2,3,-,n为节点编号.每个节点都有一个分数(均为正整数),记第i个节点的分数为di,tree及它的每个子树都有一 ...

  8. CODEVS1090 加分二叉树

    codevs1090 加分二叉树 2003年NOIP全国联赛提高组 题目描述 Description 设一个n个节点的二叉树tree的中序遍历为(l,2,3,…,n),其中数字1,2,3,…,n为节点 ...

  9. Vijos 1100 加分二叉树

    题目 1100 加分二叉树 2003年NOIP全国联赛提高组  时间限制: 1 s  空间限制: 128000 KB   题目描述 Description 设一个n个节点的二叉树tree的中序遍历为( ...

随机推荐

  1. onNewIntent调用时机

    在IntentActivity中重写下列方法:onCreate onStart onRestart  onResume  onPause onStop onDestroy  onNewIntent 一 ...

  2. Educational Codeforces Round 15 Cellular Network

    Cellular Network 题意: 给n个城市,m个加油站,要让m个加油站都覆盖n个城市,求最小的加油范围r是多少. 题解: 枚举每个城市,二分查找最近的加油站,每次更新答案即可,注意二分的时候 ...

  3. file_get_content和curl的性能比较

    今天在获取微信一张二维码图片时发现使用php中的file_get_content方式和curl方式竟然相差了50倍左右,直接晕倒!!!

  4. reduce的数目到底和哪些因素有关

      reduce的数目到底和哪些因素有关 1.我们知道map的数量和文件数.文件大小.块大小.以及split大小有关,而reduce的数量跟哪些因素有关呢? 设置mapred.tasktracker. ...

  5. JAVA 文本框、密码框、标签

    //文本框,密码框,标签 import java.awt.*; import javax.swing.*; public class Jiemian5 extends JFrame{ JPanel m ...

  6. 带节日和农历的js日历 带农历的脚本:

    <html><head><meta http-equiv="Content-Type" content="text/html; charse ...

  7. 你应该知道的jQuery技巧

    帮助提高你jQuery应用的简单小技巧. 回到顶部按钮 图片预加载 判断图片是否加载完 自动修补破损图像 Hover切换class类 禁用输入 停止正在加载的链接 toggle fade/slide ...

  8. linux ps命令(转载)

    来源:http://www.cnblogs.com/peida/archive/2012/12/19/2824418.html Linux中的ps命令是Process Status的缩写.ps命令用来 ...

  9. phpStudy Linux安装集成环境 (CentOS--7)

    phpStudy for Linux (lnmp+lamp一键安装包) phpStudy for Linux 支持Apache/Nginx/Tengine/Lighttpd, 支持php5.2/5.3 ...

  10. iOS学习笔记总结整理

    来源:http://mobile.51cto.com/iphone-386851_all.htm 学习IOS开发这对于一个初学者来说,是一件非常挠头的事情.其实学习IOS开发无外乎平时的积累与总结.下 ...