王强决定把年终奖用于购物,他把想买的物品分为两类:主件与附件,附件是从属于某个主件的,下表就是一些主件与附件的例子: 主件 附件 电脑 打印机,扫描仪 书柜 图书 书桌 台灯,文具 工作椅 无 如果要买归类为附件的物品,必须先买该附件所属的主件,且每件物品只能购买一次。 每个主件可以有 0 个、 1 个或 2 个附件。附件不再有从属于自己的附件。 王强查到了每件物品的价格(都是 10 元的整数倍),而他只有 N 元的预算。除此之外,他给每件物品规定了一个重要度,用整数 1 ~ 5 表示。他希望在花费不超过 N 元的前提下,使自己的满意度达到最大。 满意度是指所购买的每件物品的价格与重要度的乘积的总和,假设设第件物品的价格为,重要度为,共选中了件物品,编号依次为,则满意度为:。(其中 * 为乘号) 请你帮助王强计算可获得的最大的满意度。
输入描述:
输入的第 1 行,为两个正整数N,m,用一个空格隔开:(其中 N ( N从第 2 行到第 m+1 行,第 j 行给出了编号为 j-1 的物品的基本数据,每行有 3 个非负整数 v p q(其中 v 表示该物品的价格( v~ 5 ), q 表示该物品是主件还是附件。如果 q=0 ,表示该物品为主件,如果 q0 ,表示该物品为附件, q 是所属主件的编号)
输出描述:
输出一个正整数,为张强可以获得的最大的满意度。
示例1:
输入
1000 5
800 2 0
400 5 1
300 5 1
400 3 0
500 2 0
输出
2200
示例2:
输入
50 5
20 3 5
20 3 5
10 3 0
10 2 0
10 1 0
输出
130
说明
由第1行可知总钱数N为50以及希望购买的物品个数m为5;
第2和第3行的q为5,说明它们都是编号为5的物品的附件;
第4~6行的q都为0,说明它们都是主件,它们的编号依次为3~5;
所以物品的价格与重要度乘积的总和的最大值为10*1+20*3+20*3=130
代码
#include <iostream>
#include <system_error>
using namespace std;
struct wp {
int v = 0; //价格
int p = 0;//重要度
int q = 0;//主件还是附件
int vp = 0;//v*p
};
int main() {
int N, m ;
cin >> N >> m; //N 钱 m数量
N = N / 10;
int d[m];
struct wp wpp[m + 1];//主件
struct wp wpp1[m + 1];//附件1
struct wp wpp2[m + 1];//附件2
int optp[m + 1][N + 1];
int v, p, q;
for (int i = 1; i <= m; i++) {
cin >> v >> p >> q;
v = v / 10 ;
if (q == 0) {
wpp[i].v = v;
wpp[i].p = p;
wpp[i].vp = v * p;
} else if (wpp1[q].q == 0 && q != 0) {
wpp1[q].v = v;
wpp1[q].p = p;
wpp1[q].q = q;
wpp1[q].vp = v * p;
} else if (wpp2[q].q == 0 && q != 0) {
wpp2[q].v = v;
wpp2[q].p = p;
wpp1[q].q = q;
wpp2[q].vp = v * p;
}
}
// for (int i = 1; i <= m; i++) { //m数量 N 钱
// cout<<wpp[i].vp<<" "<<wpp[i].q<<endl ;
// }cout << endl;
for (int i = 0; i <= N; i++)
for (int j = 0; j <= m; j++)
optp[j][i] = 0;
for (int i = 1; i <= m; i++) {//m数量 N 钱
for (int j = 1; j <= N; j++) {
optp[i][j] = optp[i - 1][j];
if (j >= wpp[i].v &&(optp[i - 1][j - wpp[i].v] + wpp[i].vp) > optp[i ][j]) {
optp[i][j] = optp[i - 1][j - wpp[i].v] + wpp[i].vp;
}
if (j >= wpp[i].v + wpp1[i].v &&(optp[i -1][j - wpp[i].v - wpp1[i].v] + wpp[i].vp + wpp1[i].vp) > optp[i][j] ) {
optp[i][j] = optp[i - 1][j - wpp[i].v - wpp1[i].v] + wpp[i].vp + wpp1[i].vp ;
}
if (j >= wpp[i].v + wpp2[i].v &&(optp[i - 1][j - wpp[i].v - wpp2[i].v] + wpp[i].vp + wpp2[i].vp) > optp[i ][j]) {
optp[i][j] = optp[i - 1][j - wpp[i].v - wpp2[i].v] + wpp[i].vp + wpp2[i].vp ;
}
if (j >= wpp[i].v + wpp1[i].v + wpp2[i].v &&(optp[i - 1][j - wpp[i].v - wpp1[i].v - wpp2[i].v] + wpp[i].vp + wpp1[i].vp +wpp2[i].vp) > optp[i ][j] ) {
optp[i][j] = optp[i - 1][j - wpp[i].v - wpp1[i].v - wpp2[i].v] + wpp[i].vp +wpp1[i].vp + wpp2[i].vp;
}
}
}
// for (int i = 1; i <= m; i++) { //m数量 N 钱
// for (int j = 1; j <= N; j++) {
// cout << optp[i][j] * 10 << " ";
// }
// cout << endl;
// }
cout << optp[m][N] * 10;
return 0;
}
// 64 位输出请用 printf("%lld")
知识点:动态规划
类比背包问题,每次更新都是和最新的进行比较。