鱼C论坛

 找回密码
 立即注册
查看: 509|回复: 7

[已解决]有一道水题写挂了,什么情况

[复制链接]
发表于 2023-8-5 13:26:56 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能^_^

您需要 登录 才可以下载或查看,没有账号?立即注册

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。

输出格式
一行一个整数表示答案。

是的,一道水题我都能写挂。

  1. #include <bits/stdc++.h>
  2. using namespace std;

  3. int n, a[21], l[21], m[21][21], res = 0;
  4. bool md[21];

  5. bool check(int day) {
  6.         if (l[day] == 0) return 1;
  7.         for (int j = 1; j <= l[day]; ++j) {
  8.                 if (!md[m[day][j]]) {
  9.                         return 1;
  10.                 }
  11.         }
  12.         return 0;
  13. }

  14. int main() {
  15.         scanf("%d", &n);
  16.         for (int i = 1; i <= n; ++i) {
  17.                 scanf("%d", &a[i]);
  18.         }
  19.         for (int i = 1; i <= n; ++i) {
  20.                 scanf("%d", &l[i]);
  21.                 for (int j = 1; j <= l[i]; ++j) {
  22.                         scanf("%d", &m[i][j]);
  23.                 }
  24.         }
  25.         for (int f = 0; f <= (1 << n) - 1; ++f) {
  26.                 memset(md ,0, sizeof(md));
  27.                 int tmp = f, ans = 0, day = 1;
  28.         //        printf("%d\n", f);
  29.                 while (tmp) {
  30.                         if (tmp & 1) {
  31.                                 if (!check(day)) {
  32.                                         //printf("check failed:%d\n", day);
  33.                                         goto end;
  34.                                 }
  35.                                 md[day] = 1;
  36.                                 ans += a[day];
  37.                         }
  38.                         tmp >>= 1;
  39.                         ++day;
  40.                 }
  41.                 res = max(res, ans);
  42.                 end :;
  43.         }
  44.         printf("%d\n", res);
  45.     return 0;
  46. }
复制代码


写得有点类似状压,但是 8 个点寄了 6 个,不知道怎么回事,求GPT,人工解答。
最佳答案
2023-8-5 13:27:23
本帖最后由 学习编程中的Ben 于 2023-8-5 14:49 编辑

问题分析:
根据题目描述,蜗蜗每天都有一个列表,如果蜗蜗在列表中的每一天都在摸鱼的话,这一天蜗蜗就不能摸鱼。我们需要找出蜗蜗摸鱼能够获得的快乐值总和最大。

算法思路:
这是一个动态规划问题,可以使用状态压缩和位运算来解决。定义dp表示蜗蜗在第i天能够获得的最大快乐值总和。状态转移方程为:dp = max(dp[i-1], dp[j] + a),其中j是满足以下条件的所有天数:第i天没有要求,并且第j天的限制与第i天无冲突。

代码实现:
  1. #include <iostream>
  2. #include <vector>
  3. #include <algorithm>
  4. using namespace std;

  5. const int maxn = 21;
  6. int n, a[maxn], l[maxn];
  7. vector<int> m[maxn];
  8. int dp[1<<maxn];

  9. bool check(int day, int state) {
  10.     if (l[day] == 0) return true;
  11.     for (int j = 0; j < l[day]; ++j) {
  12.         int d = m[day][j];
  13.         if (state & (1 << d)) {
  14.             return false;
  15.         }
  16.     }
  17.     return true;
  18. }

  19. int main() {
  20.     cin >> n;
  21.     for (int i = 0; i < n; ++i) {
  22.         cin >> a[i];
  23.     }
  24.     for (int i = 0; i < n; ++i) {
  25.         cin >> l[i];
  26.         m[i].resize(l[i]);
  27.         for (int j = 0; j < l[i]; ++j) {
  28.             cin >> m[i][j];
  29.         }
  30.     }
  31.    
  32.     for (int state = 0; state < (1 << n); ++state) {
  33.         dp[state] = 0;
  34.     }

  35.     for (int i = 0; i < n; ++i) {
  36.         for (int state = 0; state < (1 << n); ++state) {
  37.             if (check(i, state)) {
  38.                 dp[state | (1 << i)] = max(dp[state | (1 << i)], dp[state] + a[i]);
  39.             }
  40.         }
  41.     }
  42.    
  43.     int answer = 0;
  44.     for (int state = 0; state < (1 << n); ++state) {
  45.         answer = max(answer, dp[state]);
  46.     }
  47.    
  48.     cout << answer << endl;

  49.     return 0;
  50. }
复制代码


测试样例:
输入:

  1. 3
  2. 1 2 3
  3. 1 0
  4. 0
  5. 2 1 2
复制代码

输出:

  1. 5
复制代码


解题思路解析:

该题目是一个动态规划问题,可以使用位运算进行状态压缩处理。首先定义dp表示蜗蜗在第i天能够获得的最大快乐值总和。然后使用一个整数state表示当前已经选择摸鱼的天数的集合,第i天是否摸鱼可以通过判断state的二进制表示中的第i位是否为1来确定。在状态转移时需要遍历前一天的所有状态,并找到满足条件的状态进行转移。最后得到的dp[(1<<n)-1]就是答案,表示选择了所有天数之后获得的最大快乐值总和。

希望以上解答对您有帮助!如果还有其他问题,请继续提问。

如果回答对你有帮助,请给我一个最佳答案!
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

发表于 2023-8-5 13:27:23 | 显示全部楼层    本楼为最佳答案   
本帖最后由 学习编程中的Ben 于 2023-8-5 14:49 编辑

问题分析:
根据题目描述,蜗蜗每天都有一个列表,如果蜗蜗在列表中的每一天都在摸鱼的话,这一天蜗蜗就不能摸鱼。我们需要找出蜗蜗摸鱼能够获得的快乐值总和最大。

算法思路:
这是一个动态规划问题,可以使用状态压缩和位运算来解决。定义dp表示蜗蜗在第i天能够获得的最大快乐值总和。状态转移方程为:dp = max(dp[i-1], dp[j] + a),其中j是满足以下条件的所有天数:第i天没有要求,并且第j天的限制与第i天无冲突。

代码实现:
  1. #include <iostream>
  2. #include <vector>
  3. #include <algorithm>
  4. using namespace std;

  5. const int maxn = 21;
  6. int n, a[maxn], l[maxn];
  7. vector<int> m[maxn];
  8. int dp[1<<maxn];

  9. bool check(int day, int state) {
  10.     if (l[day] == 0) return true;
  11.     for (int j = 0; j < l[day]; ++j) {
  12.         int d = m[day][j];
  13.         if (state & (1 << d)) {
  14.             return false;
  15.         }
  16.     }
  17.     return true;
  18. }

  19. int main() {
  20.     cin >> n;
  21.     for (int i = 0; i < n; ++i) {
  22.         cin >> a[i];
  23.     }
  24.     for (int i = 0; i < n; ++i) {
  25.         cin >> l[i];
  26.         m[i].resize(l[i]);
  27.         for (int j = 0; j < l[i]; ++j) {
  28.             cin >> m[i][j];
  29.         }
  30.     }
  31.    
  32.     for (int state = 0; state < (1 << n); ++state) {
  33.         dp[state] = 0;
  34.     }

  35.     for (int i = 0; i < n; ++i) {
  36.         for (int state = 0; state < (1 << n); ++state) {
  37.             if (check(i, state)) {
  38.                 dp[state | (1 << i)] = max(dp[state | (1 << i)], dp[state] + a[i]);
  39.             }
  40.         }
  41.     }
  42.    
  43.     int answer = 0;
  44.     for (int state = 0; state < (1 << n); ++state) {
  45.         answer = max(answer, dp[state]);
  46.     }
  47.    
  48.     cout << answer << endl;

  49.     return 0;
  50. }
复制代码


测试样例:
输入:

  1. 3
  2. 1 2 3
  3. 1 0
  4. 0
  5. 2 1 2
复制代码

输出:

  1. 5
复制代码


解题思路解析:

该题目是一个动态规划问题,可以使用位运算进行状态压缩处理。首先定义dp表示蜗蜗在第i天能够获得的最大快乐值总和。然后使用一个整数state表示当前已经选择摸鱼的天数的集合,第i天是否摸鱼可以通过判断state的二进制表示中的第i位是否为1来确定。在状态转移时需要遍历前一天的所有状态,并找到满足条件的状态进行转移。最后得到的dp[(1<<n)-1]就是答案,表示选择了所有天数之后获得的最大快乐值总和。

希望以上解答对您有帮助!如果还有其他问题,请继续提问。

如果回答对你有帮助,请给我一个最佳答案!
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2023-8-5 13:27:30 | 显示全部楼层
这道题目是一个经典的动态规划问题。我们可以使用状态压缩的方法来解决。

首先,我们定义一个二维数组  dp ,其中  dp[ i][j]  表示在第  i  天,状态为  j  时能够获得的最大快乐值。

然后,我们可以使用动态规划的思想来递推求解  dp  数组。具体的递推关系如下:

1. 初始化  dp[0][j]  为 0,表示第 0 天的任何状态下都无法获得快乐值。

2. 对于每一天  i ,遍历所有的状态  j :

   a. 如果第  i  天的列表为空,即  l[ i] = 0 ,那么第  i  天的状态  j  可以直接从第  i-1  天的状态  j  转移而来,即  dp[ i][j] = dp[i-1][j] + a[ i] 。
   
   b. 如果第  i  天的列表不为空,即  l[ i] > 0 ,那么我们需要判断第  i  天的状态  j  是否满足列表中的要求。具体的判断方法如下:
   
      · 首先,我们定义一个变量  valid ,初始值为  true ,表示第  i  天的状态  j  满足列表中的要求。
      
      · 然后,我们遍历第  i  天的列表中的每一天  k :
      
        · 如果第  k  天的状态  j  为 0,表示第  k  天在状态  j  下没有摸鱼,那么第  i  天的状态  j  不满足列表中的要求,我们将  valid  设置为  false 。
        
        · 如果第  k  天的状态  j  不为 0,表示第  k  天在状态  j  下摸了鱼,那么第  i  天的状态  j  满足列表中的要求。
        
      · 最后,如果  valid  为  true ,那么第  i  天的状态  j  可以从第  i-1  天的状态  j  转移而来,即  dp[ i][j] = dp[i-1][j] + a[ i] 。
      
3. 最后,我们遍历所有的状态  j ,找到能够获得的最大快乐值  max_happiness ,即  max_happiness = max(dp[n][j]) ,其中  n  表示总天数。

下面是修改后的代码:

  1. #include <bits/stdc++.h>
  2. using namespace std;

  3. int n, a[21], l[21], m[21][21], dp[21][1 << 21];

  4. int main() {
  5.     scanf("%d", &n);
  6.     for (int i = 1; i <= n; ++i) {
  7.         scanf("%d", &a[ i]);
  8.     }
  9.     for (int i = 1; i <= n; ++i) {
  10.         scanf("%d", &l[ i]);
  11.         for (int j = 1; j <= l[ i]; ++j) {
  12.             scanf("%d", &m[ i][j]);
  13.         }
  14.     }
  15.     for (int i = 1; i <= n; ++i) {
  16.         for (int j = 0; j < (1 << n); ++j) {
  17.             dp[ i][j] = dp[i-1][j] + a[ i];
  18.             for (int k = 1; k <= l[ i]; ++k) {
  19.                 if (j & (1 << (m[ i][k]-1))) {
  20.                     dp[ i][j] = max(dp[ i][j], dp[i-1][j-(1 << (m[ i][k]-1))] + a[ i]);
  21.                 }
  22.             }
  23.         }
  24.     }
  25.     int max_happiness = 0;
  26.     for (int j = 0; j < (1 << n); ++j) {
  27.         max_happiness = max(max_happiness, dp[n][j]);
  28.     }
  29.     printf("%d\n", max_happiness);
  30.     return 0;
  31. }
复制代码


希望能帮助到你!
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2023-8-5 13:35:57 | 显示全部楼层
一个爆零,一个寄 7 个点
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2023-8-5 14:50:54 | 显示全部楼层
可以给我链接吗,我写一下
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2023-8-5 14:55:40 | 显示全部楼层
链接?
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

 楼主| 发表于 2023-8-5 14:57:42 | 显示全部楼层
学习编程中的Ben 发表于 2023-8-5 14:50
可以给我链接吗,我写一下

链接不方便,这个是课程里面的题目
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2023-8-5 14:58:55 | 显示全部楼层
zhangjinxuan 发表于 2023-8-5 14:57
链接不方便,这个是课程里面的题目

emQAQ
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|鱼C工作室 ( 粤ICP备18085999号-1 | 粤公网安备 44051102000585号)

GMT+8, 2024-5-20 12:21

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表