代码随想录算法训练营第三十天 | 01背包问题 二维 01背包问题 一维 416. 分割等和子集

  1. 携带研究材料(第六期模拟笔试)
题目描述

小明是一位科学家,他需要参加一场重要的国际科学大会,以展示自己的最新研究成果。他需要带一些研究材料,但是他的行李箱空间有限。这些研究材料包括实验设备、文献资料和实验样本等等,它们各自占据不同的空间,并且具有不同的价值。

小明的行李空间为 N,问小明应该如何抉择,才能携带最大价值的研究材料,每种研究材料只能选择一次,并且只有选与不选两种选择,不能进行切割。

输入描述

第一行包含两个正整数,第一个整数 M 代表研究材料的种类,第二个正整数 N,代表小明的行李空间。

第二行包含 M 个正整数,代表每种研究材料的所占空间。

第三行包含 M 个正整数,代表每种研究材料的价值。

输出描述

输出一个整数,代表小明能够携带的研究材料的最大价值。

输入示例
复制代码
6 1
2 2 3 1 5 2
2 3 1 5 4 3
输出示例
复制代码
5
cpp 复制代码
#include<bits/stdc++.h>
using namespace std;

int n,bagweight;

void solvation()
{
    std::vector<int> weight(n,0);
    vector<int> value(n,0);
    for(int i = 0 ; i < n; i++)
    {
        cin >> weight[i];
    }
    for(int i = 0 ; i < n ; i++)
    {
        cin >> value[i];
    }
    
    vector<vector<int>> dp(weight.size(),vector<int>(bagweight+1,0));
    
    for(int i = weight[0] ; i <= bagweight ; i ++)
    {
        dp[0][i] = value[0];
    }
    
    for(int i = 1; i < weight.size() ; i++)
    {
        for(int j = 0; j <= bagweight ; j ++)
        {
            if(weight[i] > j) dp[i][j] = dp[i-1][j];
            else dp[i][j] = max(dp[i-1][j],dp[i-1][j - weight[i]]+ value[i] );
        }
    }
    
    cout << dp[weight.size() - 1][bagweight];
}


int main()
{
    while(cin >> n >> bagweight)
    {
        solvation();
    }
    
    return 0 ;
}

滚动数组的写法:

cpp 复制代码
#include<bits/stdc++.h>
#include<vector>
using namespace std;

int n ,bagweight;

void solveproblem()
{
    std::vector<int> weight(n);
    vector<int> value(n);
    
    for(int i = 0; i < n; i++)
    {
        cin >> weight[i];
    }
    for(int i = 0; i < n; i++)
    {
        cin >> value[i];
    }
    
    vector<int> dp(bagweight+1,0);
    
    for(int i = 0 ; i < weight.size(); i++)
    {
        for( int j = bagweight ; j >= weight[i] ; j--)
        {
            if(j < weight[i]) dp[j] = dp[j-1];
            else dp[j] = max(dp[j],dp[j - weight[i]] + value[i]);
        }//注意这里不是max(dp[j-1])
    }
    
    cout << dp[bagweight] <<endl;
}

int main()
{
    while(cin >> n >> bagweight)
    {
        solveproblem();
    }
    return 0 ;
}

416. 分割等和子集

给你一个 只包含正整数非空 数组 nums 。请你判断是否可以将这个数组分割成两个子集,使得两个子集的元素和相等。

示例 1:

复制代码
输入:nums = [1,5,11,5]
输出:true
解释:数组可以分割成 [1, 5, 5] 和 [11] 。

示例 2:

复制代码
输入:nums = [1,2,3,5]
输出:false
解释:数组不能分割成两个元素和相等的子集
cpp 复制代码
class Solution {
public:
    bool canPartition(vector<int>& nums) 
    {
        int sum = 0;
        vector<int> dp(20000,0);
        for(int i = 0 ; i < nums.size() ; i++)
        {
            sum += nums[i];
        }
        if(sum%2 == 1) return false;

        int mid = sum/2;

        for(int i = 0 ; i < nums.size(); i++)
        {
            for(int j = mid ; j >= nums[i] ; j--)
            {
                dp[j] = max(dp[j],dp[j - nums[i]] + nums[i]);
            }
        }
        if(dp[mid] == mid) return true;
        return false;
    }
};
相关推荐
m0_603888714 分钟前
Scaling Trends for Multi-Hop Contextual Reasoning in Mid-Scale Language Models
人工智能·算法·ai·语言模型·论文速览
Xの哲學7 分钟前
Linux io_uring 深度剖析: 重新定义高性能I/O的架构革命
linux·服务器·网络·算法·边缘计算
scx2013100412 分钟前
20260105 莫队总结
c++
comli_cn14 分钟前
残差链接(Residual Connection)
人工智能·算法
Aaron158821 分钟前
基于VU13P在人工智能高速接口传输上的应用浅析
人工智能·算法·fpga开发·硬件架构·信息与通信·信号处理·基带工程
予枫的编程笔记23 分钟前
【论文解读】DLF:以语言为核心的多模态情感分析新范式 (AAAI 2025)
人工智能·python·算法·机器学习
im_AMBER32 分钟前
Leetcode 99 删除排序链表中的重复元素 | 合并两个链表
数据结构·笔记·学习·算法·leetcode·链表
Q741_14741 分钟前
海致星图招聘 数据库内核研发实习生 一轮笔试 总结复盘(1) 作答语言:C/C++ 链表 二叉树
开发语言·c++·经验分享·面试·笔试
咔咔咔的1 小时前
1970. 你能穿过矩阵的最后一天
c++
_OP_CHEN1 小时前
【从零开始的Qt开发指南】(十九)Qt 文件操作:从 I/O 设备到文件信息,一站式掌握跨平台文件处理
开发语言·c++·qt·前端开发·文件操作·gui开发·qt文件