题目描述
学校实行学分制。每门的必修课都有固定的学分,同时还必须获得相应的选修课程学分。学校开设了N(N<300)门的选修课程,每个学生可选课程的数量M是给定的。学生选修了这M门课并考核通过就能获得相应的学分。
在选修课程中,有些课程可以直接选修,有些课程需要一定的基础知识,必须在选了其它的一些课程的基础上才能选修。例如《Frontpage》必须在选修了《Windows操作基础》之后才能选修。我们称《Windows操作基础》是《Frontpage》的先修课。每门课的直接先修课最多只有一门。两门课也可能存在相同的先修课。每门课都有一个课号,依次为1,2,3,…。 例如:
【详见图片】
表中1是2的先修课,2是3、4的先修课。如果要选3,那么1和2都一定已被选修过。 你的任务是为自己确定一个选课方案,使得你能得到的学分最多,并且必须满足先修课优先的原则。假定课程之间不存在时间上的冲突。
输入输出格式
输入格式:第一行有两个整数N,M用空格隔开。(1<=N<=200,1<=M<=150)
接下来的N行,第I+1行包含两个整数ki和si, ki表示第I门课的直接先修课,si表示第I门课的学分。若ki=0表示没有直接先修课(1<=ki<=N, 1<=si<=20)。
输出格式:
只有一行,选M门课程的最大得分。
输入输出样例
输入样例#1:
7 4 2 2 0 1 0 4 2 1 7 1 7 6 2 2
输出样例#1:
13
#include<string.h> #include<cstdio> #include<cstdlib> #include<vector> #include<iostream> #include<algorithm> using namespace std; const int MAXN=301; typedef struct { int l,r,c; }tree; tree node[MAXN]; int f[MAXN][MAXN]; int M,N; inline void mody(int i,int j) { node[i].r=node[j].l; node[j].l=i; return; } void init() { scanf("%d%d",&N,&M); int i,a,b; memset(node,-1,sizeof(node)); memset(f,-1,sizeof(f)); for(i=1;i<=N;i++) { scanf("%d%d",&a,&b); mody(i,a); node[i].c=b; } return; } int DFS(int x,int y) { if(!y||x<0) return 0; if(!x) return DFS(node[x].l,y); int i; if(f[x][y]>0) return f[x][y]; f[x][y]=DFS(node[x].r,y); for(i=1;i<=y;i++) f[x][y]=max(f[x][y],DFS(node[x].l,i-1)+node[x].c+DFS(node[x].r,y-i)); return f[x][y]; } int main() { init(); printf("%d\n",DFS(0,M)); return 0; }