【代码随想录】【算法训练营】【第42天】 [1049]最后一块石头的重量II [494]目标和 [474]一和零

前言

思路及算法思维,指路 代码随想录。
题目来自 LeetCode。

day 42,周二,坚持一下~

题目详情

[1049] 最后一块石头的重量II

题目描述

1049 最后一块石头的重量II
1049 最后一块石头的重量II

解题思路

前提:最多只会剩下一块 石头,求此石头最小的可能重量
思路:将原数组分为两个子集,使其重量和最相近,此时剩余重量最小。0-1背包问题,动态规划。
重点:问题转化为0-1背包问题,dp数组的定义、初始化及推导公式。

代码实现

C语言
dp[i][j]

分为两个子集,使其重量和最相近,此时剩余重量最小
dp[i][j]: 在[0, i]中, 不大于重量j的石头重量和
dp[i][j] = max(dp[i-1][j], dp[i-1][j-stones[i]]+stones[i])

int sumFun(int *stones, int stonesSize)
{
    int sum = 0;
    for (int i = 0; i < stonesSize; i++) {
        sum += stones[i];
    }
    return sum;
}

int maxFun(int p1, int p2)
{
    return p1 > p2 ? p1 : p2;
}

int lastStoneWeightII(int* stones, int stonesSize) {
    int totalSum = sumFun(stones, stonesSize);
    int target = totalSum / 2;
    int dp[stonesSize][target + 1];
    // 初始化dp数组
    for (int j = 0; j <= target; j++) {
        if (j < stones[0]) {
            dp[0][j] = 0;
        } else {
            dp[0][j] = stones[0];
        }
    }
    for (int i = 1; i < stonesSize; i++) {
        for (int j = 0; j <= target; j++) {
            if (j < stones[i]) {
                dp[i][j] = dp[i - 1][j];
            } else {
                dp[i][j] = maxFun(dp[i - 1][j], dp[i - 1][j -stones[i]] + stones[i]);
            }
        }
    }
    return ((totalSum - dp[stonesSize - 1][target]) - dp[stonesSize - 1][target]);
}
dp[j]

分为两个子集,使其重量和最相近,此时剩余重量最小
压缩dp[i][j]为dp[j]: 在[0, i]中, 不大于重量j的石头重量和
dp[j] = max(dp[j], dp[j-stones[i]]+stones[i])

int sumFun(int *stones, int stonesSize)
{
    int sum = 0;
    for (int i = 0; i < stonesSize; i++) {
        sum += stones[i];
    }
    return sum;
}

int maxFun(int p1, int p2)
{
    return p1 > p2 ? p1 : p2;
}

int lastStoneWeightII(int* stones, int stonesSize) {
    int totalSum = sumFun(stones, stonesSize);
    int target = totalSum / 2;
    int dp[target + 1];
    // 初始化dp数组
    for (int j = 0; j <= target; j++) {
        dp[j] = 0;
    }
    // 遍历石头,从大到小遍历target
    for (int i = 0; i < stonesSize; i++) {
        for (int j = target; j >= stones[i]; j--) {
            dp[j] = maxFun(dp[j], dp[j -stones[i]] + stones[i]);
        }
    }
    return ((totalSum - dp[target]) - dp[target]);
}

[494] 目标和

题目描述

494 目标和
494 目标和

解题思路

前提:运算结果等于 target 的数量,target = left - right ,sum = left + right
思路:推到可知 left = (target + sum) / 2,转化为数组[0, i]中数值之和恰好等于left的数目,0-1背包问题 dp[i][j] = dp[i - 1][j] + dp[i - 1][j - num[i]]
重点:问题的转化,dp[i][j] 数组的初始化,以及公式推导

代码实现

C语言
dp[i][j]

dp[i][j]数组初始化,是个大问题!!!
关注数组全0情况~

// target = left - right
// sum = left + right
// left = (target + sum) / 2
// dp[i][j]: [0, i]中数值之和q恰好=left的数目
// dp[i][j] = dp[i - 1][j] + dp[i - 1][j - num[i]]

int sumFun(int *nums, int numsSize)
{
    int totalSum = 0;
    for (int i = 0; i < numsSize; i++) {
        totalSum += nums[i];
    }
    return totalSum;
}

int findTargetSumWays(int* nums, int numsSize, int target) {
    int sum = sumFun(nums, numsSize);
    // 考虑target为负数的情况
    if ((sum < target) || ((sum + target) < 0) || ((target + sum) % 2 == 1)) {
        return 0;
    }
    int left = (target + sum) / 2;
    int dp[numsSize][left + 1];
    // dp初始化
    dp[0][0] = 1;
    for (int j = 0; j <= left; j++) {
        dp[0][j] = 0;
        if (j == 0) {
            dp[0][0] = 1;
        }
        if (j == nums[0]) {
            dp[0][j] += dp[0][j - nums[0]];
        }
    }
    // 遍历数组元素及left大小
    for (int i = 1; i < numsSize; i++) {
        for(int j = 0; j <= left; j++) {
            dp[i][j] = dp[i - 1][j];
            if (j >= nums[i]) {
                dp[i][j] += dp[i - 1][j - nums[i]];
            }
            printf("%d ", dp[i][j]);
        }
        printf("\n");
    }
    return dp[numsSize - 1][left];
}
dp[j]
// target = left - right
// sum = left + right
// right = (sum - target) / 2, 这样可以避免考虑target为很大负值的特殊情况
// 压缩为滚动一维数组dp[j]: [0, i]中数值之和恰好=right的数目
// dp[j] = dp[j] + dp[j - num[i]]

int sumFun(int *nums, int numsSize)
{
    int totalSum = 0;
    for (int i = 0; i < numsSize; i++) {
        totalSum += nums[i];
    }
    return totalSum;
}

int findTargetSumWays(int* nums, int numsSize, int target) {
    int sum = sumFun(nums, numsSize);
    if ((sum < target) || ((sum - target) % 2 == 1)) {
        return 0;
    }
    int right = (sum - target) / 2;
    int dp[right + 1];
    // dp初始化
    dp[0] = 1;
    for (int j = 1; j <= right; j++) {
        dp[j] = 0;
    }
    // 遍历数组元素, 从大到小遍历right大小
    for (int i = 0; i < numsSize; i++) {
        for(int j = right; j >= nums[i]; j--) {
            if (j >= nums[i]) {
                dp[j] += dp[j - nums[i]];
            }
        }
    }
    return dp[right];
}

[474] 一和零

题目描述

474 一和零
474 一和零

解题思路

前提:虽然有“最多”的字眼,但是对于数组strs来说,还是最多只取一个元素,所以还是0-1背包问题
思路:该背包的维度是两个,str的遍历另算,所以非滚动数组是个三维数组,为了赋值方便,直接使用滚动二维数组实现
重点:问题的区分,dp[i][j] 数组的初始化,以及公式推导

代码实现

C语言
滚动数组dp[i][j],背包的维度是两个,str的遍历另算
// 此处的dp[i][j]是滚动数组,背包的维度是两个,str的遍历另算
// dp[i][j]: 最多有i个0, j个1的strs的子集的最大长度
// dp[i][j] = max(dp[i][j], dp[i-zeroNum][j-oneNum]+1)

int maxFun(int p1, int p2)
{
    return p1 > p2 ? p1 : p2;
}

int findMaxForm(char** strs, int strsSize, int m, int n) {
    int dp[m + 1][n + 1];
    // 初始化dp
    for (int i = 0; i <= m; i++) {
        for (int j = 0; j <= n; j++) {
            dp[i][j] = 0;
        }
    }
    // 遍历strs数组, 并且从大到小遍历m和n两个维度的背包
    for (int s = 0; s < strsSize; s++) {
        // 计算该字符串的0和1个数
        int zeroNum = 0;
        int oneNum = 0;
        for (int k = 0; k < strlen(strs[s]); k++) {
            if (strs[s][k] == '0') {
                zeroNum++;
            } else {
                oneNum++;
            }
        }
        // 0-1背包, 从大到小遍历m和n两个维度的背包
        for (int i = m; i >= zeroNum; i--) {
            for (int j = n; j >= oneNum; j--) {
                dp[i][j] = maxFun(dp[i][j], dp[i - zeroNum][j - oneNum] + 1);
            }
        }
    }
    return dp[m][n];
}

今日收获

  1. 0-1背包问题:问题的转化与识别,dp数组的初始化、dp公式推导

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/720255.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

SQL Server入门-SSMS简单使用(2008R2版)-2

环境&#xff1a; win10&#xff0c;SQL Server 2008 R2 参考&#xff1a; SQL Server 管理套件&#xff08;SSMS&#xff09;_w3cschool https://www.w3cschool.cn/sqlserver/sqlserver-oe8928ks.html SQL Server存储过程_w3cschool https://www.w3cschool.cn/sqlserver/sql…

MySQL Explain 关键字详解

概述 explain 关键字可以模拟执行 sql 查询语句&#xff0c;输出执行计划&#xff0c;分析查询语句的执行性能 使用方式如下&#xff1a;explain sql explain select * from t1执行计划各字段含义 1. id 如果 id 序号相同&#xff0c;从上往下执行如果 id 序号不同&#…

Blurry - hackthebox

简介 靶机名称&#xff1a;Blurry 难度&#xff1a;中等 靶场地址&#xff1a;https://app.hackthebox.com/machines/605 本地环境 靶机IP &#xff1a;10.10.11.19 linux渗透机IP(kali 2024.2)&#xff1a;10.10.16.17 windows渗透机IP&#xff08;windows11&#xff0…

FastAdmin后台开发框架 lang 任意文件读取漏洞复现

0x01 产品简介 FastAdmin是一款基于PHPBootstrap的开源后台框架&#xff0c;专为开发者精心打造。它基于ThinkPHP和Bootstrap两大主流技术构建&#xff0c;拥有完善的权限管理系统和一键生成CRUD等强大功能。FastAdmin致力于提高开发效率&#xff0c;降低开发成本&#xff0c;…

UltraEdit电脑版下载_UltraEdit文本编辑器中文版下载_UltraEdit 2024最新版软件安装包下载附加详细安装步骤

UltraEdit中文版是一款功能强大的文本编辑器&#xff0c;几乎可以满足你所有的工作需求。使用UltraEdit文本编辑器可以操作更多记事本所不能处理的工作。如&#xff1a;基本的编辑文本、十六进制、ASCLL码、语法加亮、代码折叠、代码单词拼写检查等、C 及 VB 指令突显等,附有 H…

ICMAN触摸芯片隔空感应演示

ICMAN高性能触摸芯片之隔空感应演示 ——手掌靠近感应盘但未触摸&#xff0c;芯片即可实现隔空感应&#xff0c;指示灯亮起。 上下左右快扫、慢扫&#xff1b;反应灵敏&#xff0c;无误触发。 产品方案设计的简单&#xff0c;只需要通过调整电容&#xff0c;即可满足客户灵敏…

安全测试必备工具——SQLMap 安装及基本应用

检测SQL注入漏洞的专用工具是安全人士工具箱的必备品。Sqlmap是由python开发的用来检测与利用SQL注入漏洞的免费开源工具。它可以确定哪些参数、标头或数据元素容易受到SQL注入的影响&#xff0c;以及哪些类型的攻击是可能的。本文详细讲解这款神器的安装及使用。 1、什么是SQ…

LabVIEW版本、硬件驱动和Windows版本之间兼容性

在LabVIEW应用开发和部署过程中&#xff0c;确保LabVIEW版本、硬件驱动和Windows版本之间的一致性和兼容性至关重要。这不仅影响程序的稳定性和性能&#xff0c;还关系到项目的成功实施。本文从多角度详细分析这些因素之间的兼容性问题&#xff0c;并提供相关建议。 兼容性考虑…

Windows 安装 java 环境

搭建java开发环境 java的产品叫JDK&#xff08;java开发者工具包&#xff09;,必须安装JDK才能使用Java。 一、下载——java下载网址 二、安装 直接全部下一步就行&#xff0c;&#xff08;安装路径可以更换一下&#xff09;。 配置JAVA_HOME环境变量&#xff0c; 安装完成后…

基于 Arm 虚拟硬件实现人脸特征提取模型的部署

基于 Arm 虚拟硬件实现人脸特征提取模型的部署 文章目录 1 实验背景1.1 Arm 虚拟硬件介绍1.2 文章简介 2 实验目标3 实验前准备3.1 订阅 Arm 虚拟硬件镜像的百度智能云云服务器 BCC 实例3.2 克隆实验代码 4 实验步骤4.1 配置开发环境4.1.1 配置 CMSIS-Toolbox 环境4.1.2 配置 P…

教你一招,一键学会NAS磁盘“净身出户”的好方法!

在毕业季这个充满离别与新的开始的时刻&#xff0c;空气中似乎也弥漫着一种“断舍离”的氛围。就在这个特殊的季节里&#xff0c;我们迎来了618购物节&#xff0c;各种诱人的优惠活动如雨后春笋般涌现。铁威马618优惠不断&#xff01;T系列部分低至六折&#xff01; 在这个热闹…

MathType软件下载2024最新版_MathType官方免费下载附加详细安装步骤

MathType(数学公式编辑器)是由Design Science公司研发的一款专业的数学公式编辑工具。MathType功能非常强大&#xff0c;尤其适用于专门研究数学领域的人群使用。使用MathType让你在输入数学公式的时候能够更加的得心应手&#xff0c;各种复杂的运算符号也不在话下。 安 装 包 …

影音发烧友必入:高清先生M8 8K蓝光播放机使用体验8K播放器

影音发烧友必入&#xff1a;高清先生M8 8K蓝光播放机使用体验 高清先生在5.18成功举办新品8K蓝光播放机“M8”的发布会后&#xff0c;心心念念想尝鲜&#xff0c;于是果断下单了一台。 外形 收到货后&#xff0c;是牛皮纸包装&#xff0c;醒目的“高清先生”标识印在正面&…

【凤凰房产-注册安全分析报告-缺少轨迹的滑动条】

前言 由于网站注册入口容易被黑客攻击&#xff0c;存在如下安全问题&#xff1a; 1. 暴力破解密码&#xff0c;造成用户信息泄露 2. 短信盗刷的安全问题&#xff0c;影响业务及导致用户投诉 3. 带来经济损失&#xff0c;尤其是后付费客户&#xff0c;风险巨大&#xff0c;造…

【element 】修改下拉菜单<el-dropdown样式为横向(超简单)

最终效果&#xff1a; 原先的默认样式&#xff1a; 代码&#xff1a; 只需在<style>中添加以下样式代码即可 .horizontal-dropdown .el-dropdown-menu {display: flex; }

深入Node.js:实现网易云音乐数据自动化抓取

随着互联网技术的飞速发展&#xff0c;数据已成为企业和个人获取信息、洞察市场趋势的重要资源。音频数据&#xff0c;尤其是来自流行音乐平台如网易云音乐的数据&#xff0c;因其丰富的用户交互和内容多样性&#xff0c;成为研究用户行为和市场动态的宝贵资料。本文将深入探讨…

2 图片的分割处理和亚像素精度处理(c++和python)

本文的图片处理分为图片分割、图像的亚像素坐标处理。亚像素处理的原理可以看论文一种基于多项式插值改进的亚像素细分算法&#xff0c;该论文的详解及c的代码实现可以看博文基于多项式插值的亚像素边缘定位算法_基于多项式插值的亚像素算法-CSDN博客。下面的内容很多来自以上博…

分析医药零售数据该用哪个BI数据可视化工具?

数据是企业决策的重要依据&#xff0c;可以用于现代企业大数据可视化分析的BI工具有很多&#xff0c;各有各擅长的领域。那么哪个BI数据可视化工具分析医药零售数据又好又快&#xff1f; 做医药零售数据分析首推奥威BI数据可视化工具&#xff01; 奥威BI数据可视化工具做医药…

移动应用开发大作业报告

1 基本信息 1.1 系统名称 中华字典 1.2 开发运行环境 开发环境&#xff1a;Windows 10 专业版&#xff0c;JDK 1.8&#xff0c;AndroidStudio 运行环境&#xff1a;Java SE Runtime Environment (JRE) 8 1.3 使用的核心技术 JFrame&#xff1a;作为实现界面的窗体类&…

YOLOV8识别物体,并返回物体的像素坐标

一、YOLOV8的相关文件修改 1. 进入路径文件&#xff1a; C:\Users\82370\.conda\envs\Ayolo8\Lib\site-packages\ultralytics\engine\result.py&#xff08;此处路径为你的anacod安装的虚拟环境Ayolo8位置&#xff09; conda create -n Ayolo8 python3.11 # 虚拟环境安装代码…