选课 树形DP

摘要:
输入文件的第一行包含两个正整数M和N(用空格分隔),其中M表示要选择的课程总数(1≤M≤1000),剩余M-v[j]空间的最佳值被分配给子树,j作为递归计算的根,M]=max(F[i,M-v[j]+c[j]),v[j]<从m-v[j]]+c[j](位置j节点)中取最大值,以替换引擎的F[i,varn,0..1001];引擎的数组[0..1001];

题意/Description

    大学里实行学分。每门课程都有一定的学分,学生只要选修了这门课并考核通过就能获得相应的学分。学生最后的学分是他选修的各门课的学分的总和。 
  每个学生都要选择规定数量的课程。其中有些课程可以直接选修,有些课程需要一定的基础知识,必须在选了其它的一些课程的基础上才能选修。例如,《数据结构》必须在选修了《高级语言程序设计》之后才能选修。我们称《高级语言程序设计》是《数据结构》的先修课。每门课的直接先修课最多只有一门。两门课也可能存在相同的先修课。为便于表述每门课都有一个课号,课号依次为1,2,3,……。下面举例说明 
选课 树形DP第1张 
  上例中1是2的先修课,即如果要选修2,则1必定已被选过。同样,如果要选修3,那么1和2都一定已被选修过。 
学生不可能学完大学所开设的所有课程,因此必须在入学时选定自己要学的课程。每个学生可选课程的总数是给定的。现在请你找出一种选课方案,使得你能得到学分最多,并且必须满足先修课优先的原则。假定课程之间不存在时间上的冲突。 

 

读入/Input

    输入文件的第一行包括两个正整数M、N(中间用一个空格隔开)其中M表示待选课程总数(1≤M≤1000),N表示学生可以选的课程总数(1≤N≤M)。 
    以下M行每行代表一门课,课号依次为1,2……M。每行有两个数(用一个空格隔开),第一个数为这门课的先修课的课号(若不存在先修课则该项为0),第二个数为这门课的学分。学分是不超过10的正整数。 

 

输出/Output

    输出文件第一行只有一个数,即实际所选课程的学分总数。以下N行每行有一个数,表示学生所选课程的课号。

 

题解/solution

    这是一个树形背包,在网上look到一个O(nC)。介绍一下:

    F[i,m]表示以i为根的子树被分配到m的容量所能获得的最大得分,对于i的每一个儿子j,先将F[i,0~m-v[j]]全部赋值给F[j,0~m-v[j]],后剩下的m-v[j]个空间的最优值赋值给以j为根的子树去进行递归计算,递归计算完j后,F[i,m] = max(F[i,m],F[j,m-v[j]]+c[j]),v[j]<=m<=M。我觉得是F[j,m]的最优值包含了F[i,m]的最优值,所以只需要在F[i,m]和F[j,m-v[j]]+c[j](放j结点)中取最大值代替F[i,m]即可。大概就是这样了。

 

代码/Code

<strong>var
  n,m:longint;
  g,f:array [0..1001,0..1001] of longint;
  a:array [0..1001] of longint;
function max(t,k:longint):longint;
begin
  if t>k then exit(t);
  exit(k);
end;

procedure main(t,x:longint);
var
  i,j:longint;
begin
  if x<=0 then exit;
  for i:=1 to g[t,0] do
    begin
      for j:=0 to x-1 do
        f[g[t,i],j]:=f[t,j]+a[g[t,i]];
      main(g[t,i],x-1);
      for j:=1 to x do
        f[t,j]:=max(f[t,j],f[g[t,i],j-1]);
    end;
end;

procedure init;
var
  i,o:longint;
begin
  fillchar(f,sizeof(f),0);
  readln(n,m);
  for i:=1 to n do
    begin
      readln(o,a[i]);
      inc(g[o,0]);
      g[o,g[o,0]]:=i;
    end;
end;

begin
  init;
  main(0,m);
  write(f[0,m]);
end.
</strong>



免责声明:文章转载自《选课 树形DP》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇编译caffe的诸多注意事项Linux常用命令(1)下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

从零开始学习GDI+ (二) 基本概念与基本操作

       从零开始学习GDI+ (一)我的第一个GDI+程序        上文给新手学习GDI+讲述了vs环境等的准备工作,并且可以直接用GDI+绘图了。本文开始,讲述的可能偏理论,建议学习的过程中大胆尝试,多使用API。        首先上官方文档https://docs.microsoft.com/en-us/windows/win32/gdi...

20.6.21补数据库原理笔记

对码有部分依赖、对码有部分依赖,如果限定在非主属性。消除不好的依赖,达到2、3NF。如果进一步扩大到主属性,则消除不好的依赖后,达到BCNF, BCNF的例子复杂一些: 这个是满足3NF,但是不满足BCNF的例子,运用模式分解 分解之后,达到了BCNF, 是否存在什么副作用? 前边讲2NF、3N,给人的感觉似乎是越来越好 只有好处没有坏处 那么,3NF...

[Matlab] 短时傅里叶变换spectrogram函数

Matlab 文档:https://ww2.mathworks.cn/help/signal/ref/spectrogram.html#bultmx7-x 调用:[~,f,t,ps] = spectrogram(data,opt.window,opt.noverlap,freqRange(1):freqRange(2),sample_freq,'reass...

day9-Python学习笔记(二十二)多线程,多进程

多线程,多进程 多线程: 咱们打开的程序都一个进程。 线程是包含在进程里的。 进程里面最少有一个线程 线程之间是互相独立的 主线程 cpu是几核的,就只能同时运行几个进程 python的多线程是利用不了多核cpu的,GIL 全局解释器锁的 如果想利用多核cpu的话,就是用多进程 I0密集型任务 使用io比较多的 多线程 cpu密集型任务 多进程 能使...

Python面向对象(类和对象)

自从存在以来,Python一直是面向对象的语言。 因此,创建和使用类和对象是非常容易的。 本章将学习如何使用Python面向对象编程。如果您以前没有面向对象(OO)编程的经验,可能需要查阅介绍面向对象(OO)编程课程或至少学习一些有关教程,以便掌握基本概念。下面是面向对象编程(OOP)的一个小介绍,以帮助您快速入门学习 -OOP术语概述类 - 用于定义表示用...

jQuery Easing 动画效果扩展--使用Easing插件,让你的动画更具美感。

jQuery  Easing 是一款比较老的jQuery插件,在很多网站都有应用,尤其是在一些页面滚动、幻灯片切换等场景应用比较多。它非常小巧,且有多种动画方案供选择,使用简单,而且免费。 引入Easing js文件 该插件基于jQuery,所以需要同时引入jQuery库文件和Easing js文件。 <script type="text/javas...