题目链接:https://www.luogu.org/problemnew/show/P1040

今天考试考了一个区间DP...没错就是这个...

太蒟了真是连区间DP都不会...看了看题解也看不懂,于是请了某獴dalao给补充了一下。

在这里把自己的理解写下来,算是给一些像我一样不会区间DP的萌新们一点指引。

所谓区间dp,顾名思义就是在一段区间上的动态规划。

它既要满足dp问题的最优子结构和无后效性外,还应该符合在区间上操作的特点。我们是用小区间的最优推出大区间的最优。

通常我们是拿f[i][j]表示区间i—j。在这个题中,我们就用f[i][j]表示区间i—j的最大权值。

对于区间DP,我们通常是一层循环枚举区间的长度,一层循环枚举区间的左端点。然后进行我们需要的DP就行了。

具体对这个题的做法代码里有注释。

#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std;
int n, v[31], f[31][31], root[31][31], l, r;
void print(int l, int r)
{
if(l > r) return;
printf("%d ",root[l][r]);
print(l, root[l][r]-1);
print(root[l][r]+1,r);
}
int main()
{
//freopen("binary.in","r",stdin);
//freopen("binary.out","w",stdout);
scanf("%d",&n);
for(int i = 1; i <= n; i++)
{
scanf("%d",&v[i]);
f[i][i] = v[i];//当只有自己的时候,最大就是自己
root[i][i] = i;//root[i][j]表示在区间i—j中,以哪个点作为根得到的权值最大。
} for(int k = 2; k <= n; k++)//枚举区间大小
for(int l = 1; l+k-1 <= n; l++)//枚举区间内的端点
{
r = l+k-1;
if(f[l][r] < v[l]+f[l+1][r])
{
f[l][r] = v[l]+f[l+1][r];
root[l][r] = l;
}//右子树为空,只有左子树 的情况 if(f[l][r] < v[r]+f[l][r-1])
{
f[l][r] = v[r]+f[l][r-1];
root[l][r] = r;
}//左子树为空,只有右子树 的情况 for(int i = l+1; i <= r-1; i++)
{
if(f[l][i-1]*f[i+1][r]+v[i] > f[l][r])
{
f[l][r] = f[l][i-1]*f[i+1][r]+v[i];
root[l][r] = i;
}
}//左右子树均不为空
}//整个是在枚举在一段区间内,分别以每个点做根的情况
printf("%d\n",f[1][n]);//很明显我们所求的是1—n区间
print(1,n);//输出路径不多讲了
return 0;
}

【luogu P1040 加分二叉树】 题解的更多相关文章

  1. luogu P1040 加分二叉树

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

  2. 洛谷P1040 加分二叉树题解

    dp即可 \(f[i][j]\)表示i到j的加分 相当于区间dp了 #include<cstdio> using namespace std; int v[50]; int f[55][5 ...

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

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

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

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

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

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

  6. P1040 加分二叉树

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

  7. [洛谷P1040] 加分二叉树

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

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

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

  9. 【算法•日更•第十期】树型动态规划&区间动态规划:加分二叉树题解

    废话不多说,直接上题: 1580:加分二叉树 时间限制: 1000 ms         内存限制: 524288 KB提交数: 121     通过数: 91 [题目描述] 原题来自:NOIP 20 ...

随机推荐

  1. cloudermanager安装时database connection出现Unexpected error. Unable to verify database connection(图文详解)

    不多说,直接上干货! http://www.aboutyun.com/forum.php?mod=viewthread&tid=20455&extra=&page=2 欢迎大家 ...

  2. 0.数据结构(python语言) 基本概念 算法的代价及度量!!!

    先看思维导图: *思维导图有点简陋,本着循循渐进的思想,这小节的知识大多只做了解即可. *重点在于算法的代价及度量!!!查找资料务必弄清楚. 零.四个基本概念 问题:一个具体的需求 问题实例:针对问题 ...

  3. 使用mini-define实现前端代码的模块化管理

    这篇文章主要介绍了使用mini-define实现前端代码的模块化管理,十分不错的一篇文章,这里推荐给有需要的小伙伴. mini-define 依据require实现的简易的前端模块化框架.如果你不想花 ...

  4. Java线程同步打印ABC

    需求: 三个线程,依次打印ABCABCABC.... 方案一: 使用阻塞队列,线程1从队列1获取内容打印,线程2从队列2获取内容打印,线程3从队列3中获取内容打印.线程1把B放到队列3中,线程2把C放 ...

  5. 两个command的疑惑

    1.在cqrs模式中有command和query command  命令  没有返回值,但会更改对象的状态 query 查询  有返回值  但不会改变用户的状态,对下同而言没有副作用 2.在今天的实际 ...

  6. LOJ#2552. 「CTSC2018」假面(期望 背包)

    题意 题目链接 Sol 多年以后,我终于把这题的暴力打出来了qwq 好感动啊.. 刚开始的时候想的是: 设\(f[i][j]\)表示第\(i\)轮, 第\(j\)个人血量的期望值 转移的时候若要淦这个 ...

  7. PLC-Heart

  8. TextView来实现跑马灯的效果

    <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android=&quo ...

  9. 在crontab中执行脚本重要事项

    crontab不能成功执行shell脚本的可能原因 crond进程不存在,该进程是crontab的守护进程,它必须存在才能让crontab正常使用: 系统时间不对: 环境变量的问题:crontab执行 ...

  10. Python中深浅拷贝 垃圾回收与 super继承(六)

    1 python拷贝 深拷贝,浅拷贝 与引用三者的区别 import copy a = [1, 2, 3, 4, ['a', 'b']] #原始对象 b = a #赋值,传对象的引用 c = copy ...