【树形动态规划】【CTSC1997】选课 解题报告
CTSC1997-选课
描述
学校实行学分制。每门的必修课都有固定的学分,同时还必须获得相应的选修课程学分。学校开设了N(N<300)门的选修课程,每个学生可选课程的数量M是给定的。学生选修了这M门课并考核通过就能获得相应的学分。
在选修课程中,有些课程可以直接选修,有些课程需要一定的基础知识,必须在选了其它的一些课程的基础上才能选修。例如《Frontpage》必须在选修了《Windows操作基础》之后才能选修。我们称《Windows操作基础》是《Frontpage》的先修课。每门课的直接先修课最多只有一门。两门课也可能存在相同的先修课。每门课都有一个课号,依次为1,2,3,…。 例如:
表中1是2的先修课,2是3、4的先修课。如果要选3,那么1和2都一定已被选修过。 你的任务是为自己确定一个选课方案,使得你能得到的学分最多,并且必须满足先修课优先的原则。假定课程之间不存在时间上的冲突。
输入格式
输入文件的第一行包括两个整数N、M(中间用一个空格隔开)其中1≤N≤300,1≤M≤N。
以下N行每行代表一门课。课号依次为1,2,…,N。每行有两个数(用一个空格隔开),第一个数为这门课先修课的课号(若不存在先修课则该项为0),第二个数为这门课的学分。学分是不超过10的正整数。
输出格式
输出文件每行只有一个数。第一行是实际所选课程的学分总数。以下各行的数,表示所选课程的课号。
样例输入
样例输出
来自CTSC1997(好老的题…跟我一样大了都…)经典的树状动归,需要数据的可以给我发邮件,也可以到Vijos P1180提交.
我觉得树状动态规划常用来解决一些有传递的依赖问题.怎么说呢,有一点像背包吧,但又不相同.注意面对多叉树时,我们常常需要把其转化为二叉树来解决,根节点的儿子移到根节点的左子树上,根节点的兄弟移动到根节点的右子树上.然后什么都很方便了,状态定义状态转移都很方便了.另外需要注意的是转换后的二叉树的根节点的选定,这个十分重要,是整个动态规划的起点,我们通常用child[0]的值来作为跟节点.有关的详细的细节,这里有一份很不错的讲解ppt,我就不班门弄斧了.直接上代码!
/*
ID: ringxu97
LANG: C++
TASK: 选课
*/
#include<cstdio>
#include<cstring>
#include<iostream>
#include<cmath>
#include<cstdlib>
#include<algorithm>
#include<vector>
using namespace std;
const int maxn=+;
int n,m;
int brother[maxn],child[maxn],score[maxn];
int opt[maxn][maxn];//opt[i][j]表示在i节点选择j门课程的最优值
bool res[maxn];
void read()//读入数据
{
score[]=;
scanf("%d%d",&n,&m);
score[n+]=;
memset(child,-,sizeof(child));
memset(brother,-,sizeof(brother));
for(int i=;i<=n;++i)
{
int tmp;
scanf("%d%d",&tmp,score+i);
//左儿子右兄弟储存
brother[i]=child[tmp];
child[tmp]=i;
}
} int solve(int root,int k)
{
if(root< || k<=)return ;
if(opt[root][k]>=)return opt[root][k];
opt[root][k]=solve(brother[root],k);//不选择根节点
for(int i=;i<k;++i)
{
if(opt[root][k]<solve(brother[root],i)+solve(child[root],k-i-)+score[root])//选着根节点
opt[root][k]=solve(brother[root],i)+solve(child[root],k-i-)+score[root];
}
return opt[root][k];
}
void path(int r,int k)//寻找方案
{
int &b=brother[r],&c=child[r];
if(b> && opt[b][k]==opt[r][k])
{
res[r]=;
path(b,k);
}
else
{
for(int i=;i<k;++i)
if(opt[r][k]==solve(brother[r],i)+solve(child[r],k-i-)+score[r])
{
res[r]=;
path(b,i);
path(c,k-i-);
return;
}
}
}
void print() //打印结果和方案
{
printf("%d\n",opt[][m+]);
path(,m+);
for(int i=;i<=n;++i)if(res[i])printf("%d\n",i);
}
int main()
{
freopen("input.in", "r", stdin);
read();
debug();
memset(opt,-,sizeof(opt));
memset(res,,sizeof(res));
solve(,m+);
print();
return ;
}
【树形动态规划】【CTSC1997】选课 解题报告的更多相关文章
- 【模拟题(电子科大MaxKU)】解题报告【树形问题】【矩阵乘法】【快速幂】【数论】
目录: 1:一道简单题[树形问题](Bzoj 1827 奶牛大集会) 2:一道更简单题[矩阵乘法][快速幂] 3:最简单题[技巧] 话说这些题目的名字也是够了.... 题目: 1.一道简单题 时间1s ...
- [置顶] 刘汝佳《训练指南》动态规划::Beginner (25题)解题报告汇总
本文出自 http://blog.csdn.net/shuangde800 刘汝佳<算法竞赛入门经典-训练指南>的动态规划部分的习题Beginner 打开 这个专题一共有25题,刷完 ...
- 树形动态规划(树状DP)小结
树状动态规划定义 之所以这样命名树规,是因为树形DP的这一特殊性:没有环,dfs是不会重复,而且具有明显而又严格的层数关系.利用这一特性,我们可以很清晰地根据题目写出一个在树(型结构)上的记忆化搜索的 ...
- 【NOIP2015】提高day2解题报告
题目: P1981跳石头 描述 一年一度的“跳石头”比赛又要开始了!这项比赛将在一条笔直的河道中进行,河道中分布着一些巨大岩石.组委会已经选择好了两块岩石作为比赛起点和终点.在起点和终点之间,有 N ...
- 【第40套模拟题】【noip2011_mayan】解题报告【map】【数论】【dfs】
目录:1.潜伏者 [map] 2.Hankson的趣味题[数论]3.mayan游戏[dfs] 题目: 1. 潜伏者(spy.pas/c/cpp)[问题描述]R 国和S 国正陷入战火之中,双方都互派间谍 ...
- 2011 ACM-ICPC 成都赛区解题报告(转)
2011 ACM-ICPC 成都赛区解题报告 首先对F题出了陈题表示万分抱歉,我们都没注意到在2009哈尔滨赛区曾出过一模一样的题.其他的话,这套题还是非常不错的,除C之外的9道题都有队伍AC,最终冠 ...
- ZOJ 1093 Monkey and Banana (LIS)解题报告
ZOJ 1093 Monkey and Banana (LIS)解题报告 题目链接:http://acm.hust.edu.cn/vjudge/contest/view.action?cid= ...
- 【原创】leetCodeOj --- Sliding Window Maximum 解题报告
天,这题我已经没有底气高呼“水”了... 题目的地址: https://leetcode.com/problems/sliding-window-maximum/ 题目内容: Given an arr ...
- poj1173 解题报告
poj1173 解题报告2013-07-21 13:31 by 期待 ., 42 阅读, 0 评论, 收藏, 编辑 http://poj.org/problem?id=1173 发现此题资料甚少,斗胆 ...
随机推荐
- jquery/js当前URL对当前栏目高亮突出显示
html: 1 <div class="nav"> 2 <ul> 3 <li><a href="index.html" ...
- jquery获取元素到屏幕底的可视距离
jquery获取元素到屏幕底的可视距离 要打对号的图里的height(我自称为可视高度:滚动条未滑到最底端) 不是打叉图里的到页面底部(滚动条到最底部时的height)(offset().top方法 ...
- SSIS学习计划
百科:SSIS是Microsoft SQL Server Integration Services的简称,是生成高性能数据集成解决方案(包括数据仓库的提取.转换和加载 (ETL) 包)的平台. htt ...
- windows7在局域网中无法映射驱动器问题解决
昨天下班时闲的蛋疼,因电脑比较慢,因此在计算机的[系统配置中]的启动选项下对[启动项目]和[服务]做了误操作,导致在计算机重启之后声卡.显卡.网卡等许多服务禁用,更令人费解的是内网中断了连接,无法访问 ...
- Direct2D 加载位图
说明: 通过WIC从文件加载位图. 可缩放后加载到内存. 源码: HRESULT LoadImageFormFile( IWICImagingFactory *pWicFactory, ID2D1Re ...
- restrict和volatile的作用
每当看到这两个关键字,我都无比的头痛啊,当时看到理解了一下就明白了,但是在此遇到就忘记是怎么用的了,今天就索性写一写吧,好记性不如烂笔头呗,烂笔头不如存在网上. restrict是c99引入的,关键字 ...
- jquery 的日期时间控件(年月日时分秒)
<!-- import package --> <script type="text/javascript" src="JS/jquery.js&quo ...
- .net别样外观控件包DotNetBar
内容介绍:http://www.componentcn.com/?thread-6423-1.html BubbleBar应用: BubbleBar, DevComponents. Namesp ...
- Extjs中numberfield小数位数设置
在默认的情况下,使用numberfield控件时只会显示两位小数,有的时候需要根据业务来确定显示小数的位数.通过设置下面的属性可以达到我们想要的目的: text : '存煤量(万吨)', dataIn ...
- eval函数
eval()函数中的eval是evaluate的简称,这个函数的作用就是把一段字符串当作PHP语句来执行 <?php $a=100;eval("echo $a;"); ...