马上注册,结交更多好友,享用更多功能^_^
您需要 登录 才可以下载或查看,没有账号?立即注册
x
题目描述
蜗蜗一共有 n 天假期,在假期的第 i 天摸鱼他会得到 ai 的快乐值。如果蜗蜗每天都摸鱼的话,他会有愧疚感,所以蜗蜗制定了这么个计划:对于每一天,蜗蜗都有一个列表,如果蜗蜗在列表中的每一天都在摸鱼的话,这一天蜗蜗就不能摸鱼。现在请问蜗蜗如何摸鱼,使得他能获得的快乐值总和最大?请求出快乐值总和最大是多少。
输入格式
第一行一个整数 n 表示天数。
第二行 n 个整数 a_1,a_2,...,a_n。
接下来 n 行,对于第 i 行,先读入一个整数 l_i 表示第 i 天的列表上有几天。如果 l_i=0,也就是列表是空的话,表示第 i 天没有限制;否则,再读入 l_i 个数表示列表上列的分别是第几天,数据保证这 l_i 个数两两不同并且都不等于 i。
输出格式
一行一个整数表示答案。
是的,一道水题我都能写挂。
#include <bits/stdc++.h>
using namespace std;
int n, a[21], l[21], m[21][21], res = 0;
bool md[21];
bool check(int day) {
if (l[day] == 0) return 1;
for (int j = 1; j <= l[day]; ++j) {
if (!md[m[day][j]]) {
return 1;
}
}
return 0;
}
int main() {
scanf("%d", &n);
for (int i = 1; i <= n; ++i) {
scanf("%d", &a[i]);
}
for (int i = 1; i <= n; ++i) {
scanf("%d", &l[i]);
for (int j = 1; j <= l[i]; ++j) {
scanf("%d", &m[i][j]);
}
}
for (int f = 0; f <= (1 << n) - 1; ++f) {
memset(md ,0, sizeof(md));
int tmp = f, ans = 0, day = 1;
// printf("%d\n", f);
while (tmp) {
if (tmp & 1) {
if (!check(day)) {
//printf("check failed:%d\n", day);
goto end;
}
md[day] = 1;
ans += a[day];
}
tmp >>= 1;
++day;
}
res = max(res, ans);
end :;
}
printf("%d\n", res);
return 0;
}
写得有点类似状压,但是 8 个点寄了 6 个,不知道怎么回事,求GPT,人工解答。
本帖最后由 学习编程中的Ben 于 2023-8-5 14:49 编辑
问题分析:
根据题目描述,蜗蜗每天都有一个列表,如果蜗蜗在列表中的每一天都在摸鱼的话,这一天蜗蜗就不能摸鱼。我们需要找出蜗蜗摸鱼能够获得的快乐值总和最大。
算法思路:
这是一个动态规划问题,可以使用状态压缩和位运算来解决。定义 dp[i]表示蜗蜗在第 i天能够获得的最大快乐值总和。状态转移方程为: dp[i] = max(dp[i-1], dp[j] + a[i]),其中 j是满足以下条件的所有天数:第 i天没有要求,并且第 j天的限制与第 i天无冲突。
代码实现:#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;
const int maxn = 21;
int n, a[maxn], l[maxn];
vector<int> m[maxn];
int dp[1<<maxn];
bool check(int day, int state) {
if (l[day] == 0) return true;
for (int j = 0; j < l[day]; ++j) {
int d = m[day][j];
if (state & (1 << d)) {
return false;
}
}
return true;
}
int main() {
cin >> n;
for (int i = 0; i < n; ++i) {
cin >> a[i];
}
for (int i = 0; i < n; ++i) {
cin >> l[i];
m[i].resize(l[i]);
for (int j = 0; j < l[i]; ++j) {
cin >> m[i][j];
}
}
for (int state = 0; state < (1 << n); ++state) {
dp[state] = 0;
}
for (int i = 0; i < n; ++i) {
for (int state = 0; state < (1 << n); ++state) {
if (check(i, state)) {
dp[state | (1 << i)] = max(dp[state | (1 << i)], dp[state] + a[i]);
}
}
}
int answer = 0;
for (int state = 0; state < (1 << n); ++state) {
answer = max(answer, dp[state]);
}
cout << answer << endl;
return 0;
}
测试样例:
输入: 输出:
解题思路解析:
该题目是一个动态规划问题,可以使用位运算进行状态压缩处理。首先定义 dp[i]表示蜗蜗在第 i天能够获得的最大快乐值总和。然后使用一个整数 state表示当前已经选择摸鱼的天数的集合,第 i天是否摸鱼可以通过判断 state的二进制表示中的第 i位是否为1来确定。在状态转移时需要遍历前一天的所有状态,并找到满足条件的状态进行转移。最后得到的 dp[(1<<n)-1]就是答案,表示选择了所有天数之后获得的最大快乐值总和。
希望以上解答对您有帮助!如果还有其他问题,请继续提问。
如果回答对你有帮助,请给我一个最佳答案!
|