题目描述

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

subtreesubtree的左子树的加分× subtreesubtree的右子树的加分+subtreesubtree的根的分数。

若某个子树为空,规定其加分为11,叶子的加分就是叶节点本身的分数。不考虑它的空子树。

试求一棵符合中序遍历为(1,2,3,…,n1,2,3,…,n)且加分最高的二叉树treetree。要求输出;

(1)treetree的最高加分

(2)treetree的前序遍历

输入输出格式

输入格式:

第11行:11个整数n(n<30)n(n<30),为节点个数。

第22行:nn个用空格隔开的整数,为每个节点的分数(分数<100<100)。

输出格式:

第11行:11个整数,为最高加分(Ans \le 4,000,000,000≤4,000,000,000)。

第22行:nn个用空格隔开的整数,为该树的前序遍历。

输入输出样例

输入样例#1: 复制

5
5 7 1 2 10
输出样例#1: 复制

145
3 1 2 4 5 难以下手 看到是dfs专题一直想着用dfs来解
这题用区间dp很方便能解
f(i,j)={1 (i>j) ; 顶点i的分数 (i=j) ; max(f{i,k-1}*f{k+1,j}+顶点i的分数 (i<j) 『k取i~j』)  root[i, j]——顶点i..顶点j所组成的子树达到最大分值时的根编号。当i = j时,root[i, i] := i。
再根据树来输出
记忆化dp 
#include<bits/stdc++.h>
using namespace std;
//input by bxd
#define rep(i,a,b) for(int i=(a);i<=(b);i++)
#define repp(i,a,b) for(int i=(a);i>=(b);i--)
#define RI(n) scanf("%d",&(n))
#define RII(n,m) scanf("%d%d",&n,&m)
#define RIII(n,m,k) scanf("%d%d%d",&n,&m,&k)
#define RS(s) scanf("%s",s);
#define LL long long
#define pb push_back
#define fi first
#define REP(i,N) for(int i=0;i<(N);i++)
#define CLR(A,v) memset(A,v,sizeof A)
///////////////////////////////////
#define inf 0x3f3f3f3f
#define N 100
int root[N][N];
LL dp[N][N];
int n;
int first;
LL search1(int L,int R)//区间记忆化dp
{
if(L>R)return ;//说明为一棵空树
if(dp[L][R]==-)
{
rep(k,L,R)
{
LL cnt=search1(L,k-)*search1(k+,R)+dp[k][k];
if(cnt>dp[L][R])
{
dp[L][R]=cnt;
root[L][R]=k;
}
}
}
return dp[L][R];
} void print(int L,int R)
{
if(L>R)return ;
if(first)first=;
else printf(" ");
int x=root[L][R];
printf("%d",x);
print(L,x-);
print(x+,R);
return;
}
int main()
{
RI(n);
rep(i,,n)
rep(j,,n)
dp[i][j]=-;
rep(i,,n)
{
int x;
RI(x);
dp[i][i]=x;//每个顶点的值
root[i][i]=i;//每个点单独成一棵树 根即为自己
}
cout<<search1(,n)<<endl;
first=;
print(,n);
}

非记忆化

#include<bits/stdc++.h>
using namespace std;
//input by bxd
#define rep(i,a,b) for(int i=(a);i<=(b);i++)
#define repp(i,a,b) for(int i=(a);i>=(b);i--)
#define RI(n) scanf("%d",&(n))
#define RII(n,m) scanf("%d%d",&n,&m)
#define RIII(n,m,k) scanf("%d%d%d",&n,&m,&k)
#define RS(s) scanf("%s",s);
#define LL long long
#define pb push_back
#define fi first
#define REP(i,N) for(int i=0;i<(N);i++)
#define CLR(A,v) memset(A,v,sizeof A)
///////////////////////////////////
#define inf 0x3f3f3f3f
#define N 100
int root[N][N];
LL dp[N][N];
int n;
int first; void print(int L,int R)
{
if(L>R)return ;
if(first)first=;
else printf(" ");
int x=root[L][R];
printf("%d",x);
print(L,x-);
print(x+,R);
return;
}
int main()
{
RI(n);
rep(i,,n)
rep(j,,n)
dp[i][j]=-;
rep(i,,n)
{
int x;
RI(x);
dp[i][i]=x;//每个顶点的值
dp[i][i-]=;
root[i][i]=i;//每个点单独成一棵树 根即为自己
}
rep(len,,n)
rep(i,,n)
{
int j=i+len;
if(j<=n)
rep(k,i,j)
if(dp[i][j]<dp[i][k-]*dp[k+][j]+dp[k][k])
{
dp[i][j]=dp[i][k-]*dp[k+][j]+dp[k][k];
root[i][j]=k;
}
}
cout<<dp[][n]<<endl;
first=;
print(,n);
}

更加简洁

#include<iostream>
#include<cstdio>
using namespace std;
int n,v[],f[][],i,j,k,root[][];
void print(int l,int r){
if(l>r)return;
if(l==r){printf("%d ",l);return;}
printf("%d ",root[l][r]);
print(l,root[l][r]-);
print(root[l][r]+,r);
}
int main() {
scanf("%d",&n);
for( i=; i<=n; i++) scanf("%d",&v[i]);
for(i=; i<=n; i++) {f[i][i]=v[i];f[i][i-]=;}
for(i=n; i>=; i--)
for(j=i+; j<=n; j++)
for(k=i; k<=j; k++) {
if(f[i][j]<(f[i][k-]*f[k+][j]+f[k][k])) {
f[i][j]=f[i][k-]*f[k+][j]+f[k][k];
root[i][j]=k;
}
}
printf("%d\n",f[][n]);
print(,n);
return ;
}

P1040 加分二叉树 区间dp的更多相关文章

  1. 洛谷P1040 加分二叉树(区间dp)

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

  2. [Swust OJ 360]--加分二叉树(区间dp)

    题目链接:http://acm.swust.edu.cn/problem/360/ Time limit(ms): 1000 Memory limit(kb): 65535   Description ...

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

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

  4. 【洛谷】P1040 加分二叉树

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

  5. 【Luogu】P1040加分二叉树(区间DP)

    题目链接 区间DP,因为中序遍历的性质:区间[l,r]的任何一个数都可以是该区间的根节点. 更新权值的时候记录区间的根节点,最后DFS输出. 见代码. #include<cstdio> # ...

  6. 洛谷P1040 加分二叉树(树形dp)

    加分二叉树 时间限制: 1 Sec  内存限制: 125 MB提交: 11  解决: 7 题目描述 设一个n个节点的二叉树tree的中序遍历为(l,2,3,...,n),其中数字1,2,3,...,n ...

  7. [NOIP2003] 提高组 洛谷P1040 加分二叉树

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

  8. P1040 加分二叉树

    转自:(http://www.cnblogs.com/geek-007/p/7197439.html) 经典例题:加分二叉树(Luogu 1040) 设一个 n 个节点的二叉树 tree 的中序遍历为 ...

  9. 【luogu P1040 加分二叉树】 题解

    题目链接:https://www.luogu.org/problemnew/show/P1040 今天考试考了一个区间DP...没错就是这个... 太蒟了真是连区间DP都不会...看了看题解也看不懂, ...

随机推荐

  1. Laravel-Excel 导入 Excel 文件----为什么只获取到最后一行数据?

    ### 今天使用了Laravel-Excel 到类文件,想做一个excel  文件到导入和导出,但是看了 官方到文档示例,自己做了一下,发现 只取到到最后一行到数据, 有点摸不着头脑! 网上找了一下, ...

  2. JMS消息队列之ActiveMQ简单示例

      废话不多说,在进入主题前先看一张图,对ActiveMQ有个大体的了解:   下面进入主题:   1.添加需要的maven依赖 <!-- active mq begin --> < ...

  3. [MySQL]group by 与 having 结合函数 的统计技巧

    group by 与 having 允许字段使用函数,根据函数运行的结果group by分组或having设置选择条件; 同时group by 与 having 也允许使用字段别名 示例表a: id ...

  4. Python-Mongodb vs mysql

    mongodb https://www.cnblogs.com/kermitjam/articles/10147254.html#_label5 centos安装mongodb: https://bl ...

  5. Confluence 6 跟踪你安装中的自定义修改

    在 Confluence 中的系统信息(System Information)部分,有一个 修改(Modification)的选项.在这个选项中列出了自你 Confluence 安装以来,你 Conf ...

  6. Confluence 6 编辑一个站点装饰文件

    希望编辑一个站点的 decorator 文件: 进入  > 基本配置(General Configuration) > 布局(Layouts )(在Look and Feel 菜单下面) ...

  7. ctrl + alt + T无法启动终端

    kill -9 -1重新进入即可

  8. py4j汉语转拼音多音字处理

    先看下效果 一 .布局 <!-- 上面的搜索框 --> <com.example.editablealphalist.widgget.ClearEditText android:id ...

  9. jq插件模板

    !(function($,window, document, undefined){ function Plugin(ele,options){ //默认参数设置 this.defaults={ pa ...

  10. laravel 获取当前月,当前星期,当天起始时间方法

    获取当前月起始时间: 1. $time=time(); $start=date('Y-m-01',$time);//获取指定月份的第一天 $end=date('Y-m-t',$time); //获取指 ...