NOI2009 二叉查找树 【区间dp】
【NOI2009】二叉查找树
【问题描述】
已知一棵特殊的二叉查找树。根据定义,该二叉查找树中每个结点的数据值
都比它左子树结点的数据值大,而比它右子树结点的数据值小。
另一方面,这棵查找树中每个结点都有一个权值,每个结点的权值都比它的
儿子结点的权值要小。
已知树中所有结点的数据值各不相同;所有结点的权值也各不相同。这时可
得出这样一个有趣的结论:如果能够确定树中每个结点的数据值和权值,那么树
的形态便可以唯一确定。因为这样的一棵树可以看成是按照权值从小到大顺序插
入结点所得到的、按照数据值排序的二叉查找树。
一个结点在树中的深度定义为它到树根的距离加 1。因此树的根结点的深度
为 1。
每个结点除了数据值和权值以外,还有一个访问频度。我们定义一个结点在
树中的访问代价为它的访问频度乘以它在树中的深度。整棵树的访问代价定义为
所有结点在树中的访问代价之和。
现在给定每个结点的数据值、 权值和访问频度, 你可以根据需要修改某些
结点的权值,但每次修改你会付出 K 的额外修改代价。你可以把结点的权值改
为任何实数,但是修改后所有结点的权值必须仍保持互不相同。现在你要解决的
问题是, 整棵树的访问代价与额外修改代价的和最小是多少?
【输入格式】
输入文件为 treapmod.in。
输入文件第一行包含两个正整数 N 和 K。 N 为结点的个数, K 为每次
修改所需的额外修改代价。
接下来一行包含 N 个非负整数,是每个结点的数据值。
再接下来一行包含 N 个非负整数,是每个结点的权值。
再接下来一行包含 N 个非负整数,是每个结点的访问频度。
所有的数据值、权值、访问频度均不超过 400000。每两个数之间都有一个
空格分隔,且行尾没有空格。
【输出格式】
输出文件为 treapmod.out。
输出文件只有一个数字,即你所能得到的整棵树的访问代价与额外修改代价
之和的最小值。
【样例 1 输入】
4 10
1 2 3 4
1 2 3 4
1 2 3 4
【样例 1 输出】
29
第 6 页 共 8 页
【样例 1 说明】
输入的原图是左图,它的访问代价是 1×1+2×2+3×3+4×4=30。最佳的修改方
案 是 把 输 入 中 的 第 3 个 结 点 的 权 值 改 成 0 , 得 到 右 图 ,
访 问 代 价 是 1×2+2×3+3×1+4×2=19,加上额外修改代价 10,一共是 29。
【子任务】
40%的数据满足 N ≤ 30;
70%的数据满足 N ≤ 50;
100%的数据满足 N ≤ 70, 1 ≤ K ≤ 30000000。
【题解】
数据值已经确定了,那么这棵树的中序遍历就已经确定了,我们相当于改变权值来改变深度。
而权值可以是任意实数,但是却和答案无关,我们完全可以离散化一下。
那么我们设dp[i][j][w]表示以区间[i,j]内的节点为树,根节点都>=w的方案总数。
我们通过枚举区间内的根k,可以写出转移方程(sum(i,j)表示[i,j]的访问频率和)。
当k>=w时,我们就可以把k之间弄成根,dp[i][j][w]=min(dp[i][k-1][wk]+dp[k+1][j][wk]+sum(i,j));
也可以把k的权值修改成w,dp[i][j][w]=min(dp[i][k-1][w]+dp[k+1][w]+K+sum(i,j));
因为如果直接dp偏序关系不太明确,在实现时可以使用记忆化搜索。(表示一开始直接dp调了一个上午没调出来)
【Code】
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<cmath>
using namespace std;
#define ll long long
#define REP(i,a,b) for(register int i=(a),_end_=(b);i<=_end_;i++)
#define DREP(i,a,b) for(register int i=(a),_end_=(b);i>=_end_;i--)
#define EREP(i,a) for(register int i=start[(a)];i;i=e[i].next)
inline int read()
{
int sum=,p=;char ch=getchar();
while(!((''<=ch && ch<='') || ch=='-'))ch=getchar();
if(ch=='-')p=-,ch=getchar();
while(''<=ch && ch<='')sum=sum*+ch-,ch=getchar();
return sum*p;
} const int maxn=; int n,K; struct node {
int a,b,c;
};
node g[maxn]; bool cmp (const node x,const node y)
{
return x.a<y.a;
} int Sum(int x,int y)
{
return g[y].c-g[x-].c;
} void init()
{
int s[maxn];
n=read();K=read();
REP(i,,n)g[i].a=read();
REP(i,,n)g[i].b=read(),s[i]=g[i].b;
REP(i,,n)g[i].c=read();
sort(g+,g+n+,cmp);
sort(s+,s+n+);
int cnt=unique(s+,s+n+)-(s+);
REP(i,,n)
{
REP(j,,cnt)
{
if(g[i].b==s[j]){g[i].b=j;break;}
}
}
REP(i,,n)g[i].c+=g[i-].c;
} ll dp[maxn][maxn][maxn]; void chkmin(ll &a,ll b)
{
if(a>b)a=b;
}
#define inf 0x7ffffffffff
ll dfs(int l,int r,int k)
{
if(l>r)return ;
if(~dp[l][r][k])return dp[l][r][k];
ll ans=inf;
REP(i,l,r)
{
chkmin(ans,dfs(l,i-,k)+dfs(i+,r,k)+K);
if(g[i].b>=k)chkmin(ans,dfs(l,i-,g[i].b)+dfs(i+,r,g[i].b));
}
return dp[l][r][k]=ans+Sum(l,r);
} void doing()
{
memset(dp,-,sizeof(dp));
//REP(i,1,n)dp[i][i][g[i].b]=Sum(i,i);
/*REP(len,1,n-1)
{
REP(i,1,n-len)
{
int j=i+len;
REP(k,1,Min(i,j))
{
REP(l,i,j)
{
int Mn=Min(i,j),z=0;
if(Mn==g[l].b)z=0;else z=K;
chkmin(dp[i][j][k],dp[i][l-1][Mn]+dp[l+1][j][Mn]+z+Sum(i,j));
}
}
}
}*/
cout<<dfs(,n,)<<endl;
}
int main()
{
init();
doing();
return ;
}
NOI2009 二叉查找树 【区间dp】的更多相关文章
- bzoj 1564 [NOI2009]二叉查找树 区间DP
[NOI2009]二叉查找树 Time Limit: 10 Sec Memory Limit: 64 MBSubmit: 906 Solved: 630[Submit][Status][Discu ...
- 洛谷$P1864\ [NOI2009]$二叉查找树 区间$dp$
正解:区间$dp$ 解题报告: 传送门$QwQ$ 首先根据二叉查找树的定义可知,数据确定了,这棵树的中序遍历就已经改变了,唯一能改变的就是通过改变权值从而改变结点的深度. 发现这里权值的值没有意义,所 ...
- [BZOJ1564][NOI2009]二叉查找树 树形dp 区间dp
1564: [NOI2009]二叉查找树 Time Limit: 10 Sec Memory Limit: 64 MBSubmit: 879 Solved: 612[Submit][Status] ...
- BZOJ 1564: [NOI2009]二叉查找树( dp )
树的中序遍历是唯一的. 按照数据值处理出中序遍历后, dp(l, r, v)表示[l, r]组成的树, 树的所有节点的权值≥v的最小代价(离散化权值). 枚举m为根(p表示访问频率): 修改m的权值 ...
- [洛谷P1864] NOI2009 二叉查找树
问题描述 已知一棵特殊的二叉查找树.根据定义,该二叉查找树中每个结点的数据值都比它左儿子结点的数据值大,而比它右儿子结点的数据值小. 另一方面,这棵查找树中每个结点都有一个权值,每个结点的权值都比它的 ...
- P1864 [NOI2009]二叉查找树
链接P1864 [NOI2009]二叉查找树 这题还是蛮难的--是我菜. 题目描述中的一大堆其实就是在描述\(treap.\),考虑\(treap\)的一些性质: 首先不管怎么转,中序遍历是确定的,所 ...
- 【BZOJ-4380】Myjnie 区间DP
4380: [POI2015]Myjnie Time Limit: 40 Sec Memory Limit: 256 MBSec Special JudgeSubmit: 162 Solved: ...
- 【POJ-1390】Blocks 区间DP
Blocks Time Limit: 5000MS Memory Limit: 65536K Total Submissions: 5252 Accepted: 2165 Descriptio ...
- 区间DP LightOJ 1422 Halloween Costumes
http://lightoj.com/volume_showproblem.php?problem=1422 做的第一道区间DP的题目,试水. 参考解题报告: http://www.cnblogs.c ...
- BZOJ1055: [HAOI2008]玩具取名[区间DP]
1055: [HAOI2008]玩具取名 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 1588 Solved: 925[Submit][Statu ...
随机推荐
- linux常用命令(CentOS)
1.目录切换命令 linux目录结构 ps:绿色标注为常用命令 cd xx 切换到该目录下的xx目录 cd ../ 切换到上一层目录 cd / 切换到系统根目录 cd ~ 切换到用户主目录 cd - ...
- 【开发技术】对文件内容进行加密-java
http://hi.baidu.com/java0804ms/item/111ea834fbd4d2f596f88d5a 实现效果:对文件内容进行加密,使之直接打开成为乱码,不以明文显示 实现步骤:1 ...
- servlet入门学习之Web容器
一.web服务器 web服务器(程序/软件) 即:www服务器或http服务器.提供Web信息浏览服务.它只需支持HTTP协议.HTML文档格式及URL.向浏览器提供服务的程序. 1.Web服务器是指 ...
- scss 初学笔记 二 混合宏
混合宏 格式 @mixin 定义混合宏 (相当于变量声明 var $ ?) //不带参数混合宏 @mixin borderRadius{ -webkit-border-radius: 5px; b ...
- python_为被装饰的函数保留元数据
案例: 在函数对象中保存着一些函数的元数据,如: f.__name__ 函数名 f.__doc__ 函数文档 f.__moudle__ 函数所 ...
- 深入分析java传参
概述 java中的参数传递问题可以根据参数的类型大致可以分为三类:传递基本类型,传递String类型,传递引用类型,至于最终是否可以归纳为值传递和引用传递,根据每个人的理解不同,答案不同,此 ...
- mysql关于char和varchar的查询效率问题
看了好多资料都说 varchar(size) 可变长度的字符值,节省空间,查询效率低 char(size) 固定长度的字符值,浪费空间,查询效率高 但是实际测试 char(100) varcha ...
- 如何用命令将本地项目上传到github
一.Git终端软件安装 1.下载windows上git终端,类似shell工具,下载地址:http://msysgit.github.io/ 2.安装方法,打开文件,一路点击Next即可 3.安装完成 ...
- nodejs爬虫笔记(一)---request与cheerio等模块的应用
目标:爬取慕课网里面一个教程的视频信息,并将其存入mysql数据库.以http://www.imooc.com/learn/857为例. 一.工具 1.安装nodejs:(操作系统环境:WiN 7 6 ...
- python-day2数据类型
内容介绍 数据类型 字符编码 文件处理 1.什么是数据? x=10 , 10是我们要存储的数据. 2.为何数据要分不同的类型 数据是用来表示状态的,不同的状态就应该用不同的类型的数据去表示 3.数据类 ...