愚蠢的矿工--树形动态规划
来源:互联网 发布:怎么解除wifi网络限制 编辑:程序博客网 时间:2024/06/09 22:49
http://www.rqnoj.cn/Problem_30.html
一、思路
题目是多叉树,原本想
0 0 0 3 =0 1 2 =0 2 1 = 0 3 1 但是要列举的情况太多,就采用转换为二叉树的方法
1 2 3
1、读进多叉树tree[1001][1001]
tree[i][0]来记录本节点的孩子个数
tree[i][j]表示节点i的第j个子节点的编号
2、转化为二叉树binaryTree[1001][1001]
递归生成
先将第一个节点放到左孩子,并生成该节点的根子树
将兄弟节点放到当前生成节点的右孩子处,并生成该节点的根子树
3、记忆化搜索求解
分为三种情况
(1)只取根节点
(2)左孩子取i 右孩子取m-i-1 根节点取1 其中i [0, m-1]
(3)只取右孩子,因为是兄弟节点,所以根节点可以不留人,右孩子取m个人 根节点 和 左孩子都取0个人
取其中最大值
4、问题
第一次写完之后,超时一个点,错误结果一个点
看题解,将cin读取方式改为了scanf,时间缩短了很多
AC了
效率还是很明显的
5、代码如下:
#include <iostream.h>#include <fstream.h>int tree[1001][1001], binaryTree[1001][1001], dpTree[1001][1001];int value[1001];void ConvertToBinaryTree(int root){int currentRoot, i;if (!tree[root][0]) {return;}currentRoot = binaryTree[root][1] = tree[root][1];ConvertToBinaryTree(currentRoot);for (i=2; i<=tree[root][0]; i++){currentRoot = binaryTree[currentRoot][2] = tree[root][i];ConvertToBinaryTree(currentRoot);}}void PrintTree(int t[1001][1001], int n){int i, j;for (i=0; i<=n; i++){for (j=1; j<=t[i][0]; j++){cout<<' '<<t[i][j];}cout<<endl;}}void PrintBinaryTree(int root){if (root==0) return;cout<<root<<endl;//if (binaryTree[root][1]){}PrintBinaryTree(binaryTree[root][1]);PrintBinaryTree(binaryTree[root][2]);}int DPTree(int root, int m){int i;if (dpTree[root][m]){return dpTree[root][m];}if (m==0||root==0){return 0;}dpTree[root][m] = value[root];for (i=0; i<=m-1; i++){if (dpTree[root][m] < DPTree(binaryTree[root][1], i)+value[root]+DPTree(binaryTree[root][2], m-1-i)){dpTree[root][m] = dpTree[binaryTree[root][1]][i] + value[root] + dpTree[binaryTree[root][2]][m-1-i];}}if (dpTree[root][m] < DPTree(binaryTree[root][2], m)){dpTree[root][m] = dpTree[binaryTree[root][2]][m];}return dpTree[root][m];}int main(){int n, m;//ifstream inFile("e:\\test.txt");int i;//cin>>n>>m; scanf("%d%d", &n, &m); //inFile>>n>>m;//读进财富值for (i=1; i<=n; i++){//cin>>value[i]; scanf("%d", &value[i]); //inFile>>value[i];}//读进多叉树for (i=1; i<=n; i++){int from, to;//cin>>from>>to; scanf("%d%d", &from, &to);//inFile>>from>>to;tree[from][0]++;tree[from][tree[from][0]] = to;}//多叉树转换为二叉树ConvertToBinaryTree(0);//打印树//PrintTree(tree, n);//PrintBinaryTree(binaryTree[0][1]);//树形动规最优值cout<<DPTree(binaryTree[0][1], m);//inFile.close();return 0;}
- 愚蠢的矿工--树形动态规划
- rnqoj-30- [stupid]愚蠢的矿工-树形DP
- RQNOJ愚蠢的矿工
- [codevs1418]愚蠢的矿工
- 动态规划:矿工模型
- 【BZOJ1806】【IOI2007】Miners 矿工配餐 动态规划
- 树形动态规划 ~~没有上司的晚会
- 没有上司的晚会 树形动态规划
- 树形动态规划
- 树形动态规划
- 树形动态规划
- 选课 树形动态规划
- 将功补过 树形动态规划
- 电子眼 树形动态规划
- 树形动态规划总结
- 树形动态规划总结
- 树形动态规划
- 树形动态规划
- JSR-303规范,Bean Validation
- 书写格式
- jQuery局部刷新的实现
- Expect安装方法
- MVVM 的使用
- 愚蠢的矿工--树形动态规划
- ASP.net 类库中获得当前路径的方法(非页面中)
- hdu 1285——确定比赛名次(拓扑排序)
- WinForm中,Button按钮有PerformClick()方法,可以模拟用户单击鼠标. 即button.PerformClick()生成按钮的事件。
- 最小公倍数
- SQL Server之分布式事务.
- Service生命周期
- Nutch1.0和eclipse配置(详细版)
- 由数据表生成hibernate映射文件和实体类