SDUT_2015寒假集训_背包_G-小P寻宝记——好基友一起走

本文探讨如何运用动态规划算法解决一个寻宝过程中遇到的背包问题,通过实例解析,帮助读者理解如何在有限空间内最大化价值。面对困境,友情与智慧并存,共同寻找解决方案。

题目描述

话说,上次小P到伊利哇呀国旅行得到了一批宝藏。他是相当开心啊,回来就告诉了他的好基友小鑫,于是他们又结伴去伊利哇呀国寻宝。
这次小P的寻宝之路可没有那么的轻松,他们走到了一个森林,小鑫一不小心被触发了机关,被困在了一个大笼子里面,笼子旁边上有一道题目和一个密码锁,上面说只要解出此题输入密码即可救出被困人。小鑫不是很聪明,所以他做不出来,他知道小P很笨,更解不出来。所以他就让小P独自回去,不用管他。但是小P重情重义不会抛弃他离去。他说:“不,好基友一起走!”。于是就感动了上帝,上帝特派你来替他们解决问题。聪明的你要加油了啊!
题目描述:给你n种物品和一个体积为v的包包。每种物品有无数种,体积是vi价值是wi。求出包包v所能装的最大价值的东西。

输入

多组输入。第一行有两个正整数n(0<n<=100), v(0<v<= 10000)。接下来两行每行有n个数字。第一行表示每种物品的价值wi(0<wi<100),第二行表示每种物品的体积vi(0<vi<100)。

输出

输出最多可以得到的价值。输出结果救出小鑫。

示例输入

5 20
1 2 3 4 5
2 6 3 5 4

示例输出

25

一开始打算用二维数组做,结果做了半天结果就是不对= =无奈又改成了一维数组的做法。建立一个数组dp[j],表示背包容量为j时可得的最大价值。由题意我们可以得到状态转移方程dp[j]=max(dp[j],dp[j-v[i]]+w[i]);最终dp[v]就是我们要的答案,代码如下。

代码

#include <iostream>
#include <memory.h>

using namespace std;

int dp[10100],wi[10100],vi[10100];

int main()
{
    ios::sync_with_stdio(false);
    int n,v,i,j;
    while(cin>>n>>v)
    {
        memset(dp,0,sizeof(dp));
        for(i=1;i<=n;i++)
            cin>>wi[i];
        for(i=1;i<=n;i++)
            cin>>vi[i];
        for(i=1;i<=n;i++)
        {
            for(j=0;j<=v;j++)
            {
                if(j>=vi[i])
                    dp[j]=max(dp[j],dp[j-vi[i]]+wi[i]);
            }
        }
        cout<<dp[v]<<endl;
    }
    return 0;
}



你提到的题目是: > **7-103 sdut-C语言实验——整除** 这是山东理工大学(SDUT)在线判题系统中的一道经典 C 语言入门题,考察的是 **条件判断与整除运算**。 --- ## ✅ 题目描述(根据常见版本还原) > 给定一个整数 n,判断它是否能被 3 整除、被 5 整除、被 7 整除。 > > 输出对应的提示信息。 > > - 如果能被 3 整除,输出 `3|n` > - 如果能被 5 整除,输出 `5|n` > - 如果能被 7 整除,输出 `7|n` > > 每个成立的情况都要输出一行。如果都不能整除,则输出 `n` 不能被 3、5、7 整除。 ### 🔍 输入格式: 一个整数 `n` ### 🔍 输出格式: 按上述规则输出 ### 📌 示例 1: ``` 输入:15 输出: 3|15 5|15 ``` ### 📌 示例 2: ``` 输入:14 输出: 7|14 ``` ### 📌 示例 3: ``` 输入:11 输出: 11 cannot be divisible by 3, 5 and 7. ``` --- ## ✅ 正确解答代码(C语言) ```c #include <stdio.h> int main() { int n; scanf("%d", &n); int flag = 0; // 标记是否有能整除的情况 if (n % 3 == 0) { printf("3|%d\n", n); flag = 1; } if (n % 5 == 0) { printf("5|%d\n", n); flag = 1; } if (n % 7 == 0) { printf("7|%d\n", n); flag = 1; } if (!flag) { printf("%d cannot be divisible by 3, 5 and 7.\n", n); } return 0; } ``` --- ## 🔍 代码解释 | 代码片段 | 功能说明 | |--------|---------| | `scanf("%d", &n)` | 读入一个整数 | | `n % 3 == 0` | 判断 n 是否能被 3 整除(余数为 0) | | `flag` 变量 | 记录是否至少有一个因子能整除 n,用于控制最后的“都不能整除”输出 | | 多个独立的 `if` | 因为要检查所有三种情况,不能用 `else if`,否则只能输出一个结果 | > ⚠️ 注意:必须使用多个 `if`,而不是 `else if`,因为一个数可能同时被多个数整除(如 15 被 3 和 5 同时整除) --- ## ❗ 常见错误 | 错误 | 后果 | |------|------| | 使用 `else if` | 只输出第一个匹配项,漏掉其他整除关系 | | 忘记 `\n` 换行 | 格式错误(PE) | | 输出拼写错误,比如 `"divisable"` | 答案错误(WA) | | 没处理全都不整除的情况 | 缺少输出,判错 | --- ## ✅ 进阶优化(可选) 你可以把判断封装成宏或循环(虽然本题简单没必要): ```c // 示例:用数组简化(适合更多因子时) int divisors[] = {3, 5, 7}; for (int i = 0; i < 3; i++) { if (n % divisors[i] == 0) { printf("%d|%d\n", divisors[i], n); flag = 1; } } ``` 但对初学者来说,直接写三个 `if` 更清晰。 --- ## ✅ 总结 这道题考察: - `%` 运算符(取模) - 条件判断 `if` - 多分支独立判断(不是互斥) - 输出格式控制 - 使用标志位 `flag` 处理“无匹配”情况 是典型的“基础但易错”的实验题。 --- ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值