LeetCode377. 组合总和 Ⅳ

377. 组合总和 Ⅳ

文章目录

    • [377. 组合总和 Ⅳ](https://leetcode.cn/problems/combination-sum-iv/)
      • 一、题目
      • 二、题解
        • 方法一:完全背包一维数组
          • 动态规划思路
          • 代码分析
        • 方法二:动态规划二维数组


一、题目

给你一个由 不同 整数组成的数组 nums ,和一个目标整数 target 。请你从 nums 中找出并返回总和为 target 的元素组合的个数。

题目数据保证答案符合 32 位整数范围。

示例 1:

输入:nums = [1,2,3], target = 4
输出:7
解释:
所有可能的组合为:
(1, 1, 1, 1)
(1, 1, 2)
(1, 2, 1)
(1, 3)
(2, 1, 1)
(2, 2)
(3, 1)
请注意,顺序不同的序列被视作不同的组合。

示例 2:

输入:nums = [9], target = 3
输出:0

提示:

  • 1 <= nums.length <= 200
  • 1 <= nums[i] <= 1000
  • nums 中的所有元素 互不相同
  • 1 <= target <= 1000

进阶:如果给定的数组中含有负数会发生什么?问题会产生何种变化?如果允许负数出现,需要向题目中添加哪些限制条件?

二、题解

这道题要求我们找出由给定数组 nums 中的不同元素组成的总和等于 target 的组合个数,说是组合,其实实质上是排列,我在这篇文章里讲述了关于排列组合的区别并且手写了动态规划过程:https://blog.csdn.net/m0_61843614/article/details/132745696。

方法一:完全背包一维数组

动态规划思路

定义一个一维数组 dp,其中 dp[i] 表示总和为 i 的组合个数。

我们的目标是计算 dp[target],也就是总和为 target 的组合个数。为了计算 dp[target],可以考虑如下的思路:

  1. 初始化一个长度为 target + 1 的数组 dp,并将其所有元素初始化为0。dp[i] 表示总和为 i 的组合个数。

  2. 由于组合的元素可以重复使用,我们可以遍历数组 nums 中的每个元素,并尝试将其加入到总和为 i 的组合中。

  3. 对于每个元素 nums[j],我们可以检查 dp[i - nums[j]],它表示总和为 i - nums[j] 的组合个数。我们可以将 dp[i - nums[j]] 加到 dp[i] 上,表示将 nums[j] 加入到当前组合中。

  4. 重复上述步骤,直到遍历完数组 nums 中的所有元素。

  5. 最终,dp[target] 就是我们要求的答案,表示总和为 target 的组合个数。

代码分析
class Solution {
public:int combinationSum4(vector<int>& nums, int target) {vector<long long> dp(target + 1, 0);dp[0] = 1;  // 初始化,总和为0的组合个数为1for (int i = 0; i <= target; i++) {for (int j = 0; j < nums.size(); j++) {if (nums[j] <= i && dp[i] < INT_MAX - dp[i - nums[j]]) {dp[i] = dp[i] + dp[i - nums[j]];  // 更新 dp[i]}}}return dp[target];}
};

现在我逐步解释代码的各个部分:

  • 我们定义了一个一维数组 dp,长度为 target + 1,并将所有元素初始化为0。

  • 初始化 dp[0] 为1,因为总和为0的组合只有一种方式,就是什么都不选。

  • 使用两个嵌套的循环,外层循环遍历所有可能的总和 i,内层循环遍历数组 nums 中的所有元素。

  • 在内层循环中,我们检查当前元素 nums[j] 是否小于等于 i,如果是,就说明可以将 nums[j] 加入到总和为 i 的组合中。

  • 如果 dp[i] 的值还没有越界(小于 INT_MAX - dp[i - nums[j]]),则将 dp[i - nums[j]] 的值加到 dp[i] 上,表示将 nums[j] 加入到当前组合中。

  • 最终,返回 dp[target],即总和为 target 的组合个数。

方法二:动态规划二维数组

先给出代码:

class Solution {
public:int combinationSum4(vector<int>& nums, int target) {//dp[j][i]意义是背包容量为i的情况下,最后一个加入的数字是从nums[0]到nums[i]之间的方法的总数vector<vector<long long>> dp(nums.size(), vector<long long>(target + 1, 0));dp[0][0] = 1;for (int i = 0; i <= target; i++) {for (int j = 0; j < nums.size(); j++) {if (j > 0) {dp[j][i] = dp[j - 1][i];}if (i >= nums[j] && dp[j][i] < INT_MAX - dp[nums.size() - 1][i - nums[j]]) {dp[j][i] += dp[nums.size() - 1][i - nums[j]];}}}return dp[nums.size() - 1][target];}
};
  1. 动态规划思路

这个问题的目标是找出总和为 target 的元素组合的个数。首先,让我们定义一个二维数组 dp,其中 dp[j][i]意义是背包容量为i的情况下,最后一个加入的数字是从nums[0]nums[i]之间的方法的总数。我们的目标是求 dp[nums.size() - 1][target],即使用所有的元素构成和为 target 的排列的个数。

  1. 初始化

首先,我们初始化 dp 数组,将所有元素都初始化为 0。然后我们设置 dp[0][0] = 1,这是因为在前 0 个元素中,构成和为 0 的组合有一种方式,即不选择任何元素。

  1. 填充动态规划数组

接下来,我们使用两个嵌套循环来填充 dp 数组。外层循环 i 表示考虑前 i 个元素,内层循环 j 表示目标和为 j

  • 如果 j < nums[i],意味着当前的元素 nums[i] 太大,不能加入组合中,所以我们将 dp[i][j] 设置为 dp[i-1][j],表示不选择当前元素时的组合数,继承上一行的值。

  • 如果 j >= nums[i],意味着当前的元素 nums[i] 可以加入组合中。我们需要考虑两种情况:

    • 不选择当前元素,即 dp[i][j] = dp[i-1][j]
    • 选择当前元素,即 dp[i][j] += dp[i][j - nums[i]],这里的 dp[i][j - nums[i]] 表示在考虑前 i 个元素,和为 j - nums[i] 的组合数。

最终,dp[nums.size() - 1][target] 就代表了使用所有元素构成和为 target 的组合的个数。

  1. 返回结果

最后,我们返回 dp[nums.size() - 1][target] 即可得到答案。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Java8的Stream用法

Java8 API新增了一个新的抽象流Stream&#xff0c;它可以执行非常复杂的查找、过滤和映射数据等操作。使用Stream API 对集合数据进行操作&#xff0c;就类似于使用 SQL 执行的数据库查询。Stream就是把集合数据看作流&#xff0c;流在管道中传输&#xff0c;我们可在管道中进行…

Spring MVC:请求转发与请求重定向

Spring MVC 请求转发请求重定向附 请求转发 转发&#xff08; forward &#xff09;&#xff0c;指服务器接收请求后&#xff0c;从一个资源跳转到另一个资源中。请求转发是一次请求&#xff0c;不会改变浏览器的请求地址。 简单示例&#xff1a; 1.通过 String 类型的返回值…

GO语言篇之反射

GO语言篇之反射 文章目录 GO语言篇之反射前言获取变量类型获取变量值获取结构体的字段&#xff0c;方法&#xff0c;动态地修改&#xff0c;调用结构体的字段和方法创建变量缺点 前言 Go语言可以在运行期间查看自身结构&#xff0c;在运行时动态地获取结构体的信息&#xff0c…

人工智能训练?量子计算机?显卡!【为什么人工智能需要强大的显卡而不是处理器?量子计算机可以吗?】

人工智能硬件与量子计算机&#xff1a;未来的竞争与合作 人工智能&#xff08;AI&#xff09;领域的发展一直以来都与硬件技术密不可分。随着深度学习和神经网络的崛起&#xff0c;图形处理单元&#xff08;GPU&#xff09;等硬件成为了AI训练的主要工具。然而&#xff0c;量子…

spring 理解

仅供个人学习&#xff0c;部分转自路人甲&#xff0c;侵删 spring容器 1.基本概念 2.Spring Ioc 容器 3.Spring Aop 4.数据访问 5.Spring MVC 6.事务管理 7.高级特性 8.整合其他框架 9.理解原理 基本概念 spring启动流程&#xff0c;加载配置文件&#xff0c;创建…

2023年基因编辑行业研究报告

第一章 行业发展概况 1.1 定义 基因编辑&#xff08;Gene Editing&#xff09;&#xff0c;又称基因组编辑&#xff08;Genome Editing&#xff09;或基因组工程&#xff08;Genome Engineering&#xff09;&#xff0c;是一项精确的科学技术&#xff0c;可以对含有遗传信息的…

[JAVAee]Spring的基础介绍

本文章介绍了Spring大致是什么,核心的功能. Spring是什么? Spring指的是Spring Framework(Spring框架). 支持广大的开发场景,能够使应用开发变得简单. 其集成了各种工具,还实现了底层的类的实例化和生命周期的管理. 简单来说,Spring就是拥有众多工具方法的IoC容器 容器?…

常用百宝箱——日志处理

目录 前言 一、logging库 二、logging日志等级 三、logging四大组件 四、封装示例 总结 前言 日志是记录特定时间段或事件的详细信息的文件或记录。它们通过时间戳和关键词或描述符来标识事件或行动。日志可以用于许多目的&#xff0c;例如&#xff1a;故障排除、网络安全…

搭建springcloud注册中心eureka以及admin监控

写该篇文章的目的是为了以后搭建微服务的时候避免踩坑 要求&#xff1a;搭建一个eureka-server注册中心&#xff0c;再构建两个eureka-client注册上去&#xff0c;然后再搭建admin服务注册到注册中心。实现在admin后管页面可观察已注册上去的服务 前提&#xff1a;使用的spri…

Redis 初识与入门

1. 什么是Redis Redis 是一种基于内存的数据库&#xff0c;对数据的读写操作都是在内存中完成&#xff0c;因此读写速度非常快&#xff0c;常用于缓存&#xff0c;消息队列、分布式锁等场景。 Redis 提供了多种数据类型来支持不同的业务场景&#xff0c;比如 String(字符串)、…

关系的定义及表示

关系的定义及表示 1、若集合R是AA的子集&#xff0c;则称R是集合A上的二元关系&#xff0c;简称关系 例&#xff1a;A{1,2}&#xff0c; AA{<1,1>,<1,2>,<2,1>,<2,2>}&#xff0c;AA的任何一个子集都是A上的关系 如&#xff1a; R{<1,1>, &…

java企业数据管理系统

项目介绍 此项目为企业数据管理系统的后端部分&#xff0c;前端部分请参考vue-admin&#xff0c;项目实现了菜单管理、用户管理、角色管理和权限管理四个基础模块&#xff0c;前端菜单管理结合动态路由可自由添加菜单。结合Shiro权限管理实现了菜单和按钮的权限控制。 ❝ 前端…

贝塞尔曲线的一些资料收集

一本免费的在线书籍&#xff0c;供你在非常需要了解如何处理贝塞尔相关的事情。 https://pomax.github.io/bezierinfo/zh-CN/index.html An algorithm to find bounding box of closed bezier curves? - Stack Overflow https://stackoverflow.com/questions/2587751/an-algo…

歌曲推荐《最佳损友》

最佳损友 陈奕迅演唱歌曲 《最佳损友》是陈奕迅演唱的一首粤语歌曲&#xff0c;由黄伟文作词&#xff0c;Eric Kwok&#xff08;郭伟亮&#xff09;作曲。收录于专辑《Life Continues》中&#xff0c;发行于2006年6月15日。 2006年12月26日&#xff0c;该曲获得2006香港新城…

Python之OS模块

os模块负责程序与操作系统的交互&#xff0c;提供了访问操作系统底层的接口;即os模块提供了非常丰富的方法用来处理文件和目录。 使用的时候需要导入该模块:import os

MojoTween:使用「Burst、Jobs、Collections、Mathematics」优化实现的Unity顶级「Tween动画引擎」

MojoTween是一个令人惊叹的Tween动画引擎&#xff0c;针对C#和Unity进行了高度优化&#xff0c;使用了Burst、Jobs、Collections、Mathematics等新技术编码。 MojoTween提供了一套完整的解决方案&#xff0c;将Tween动画应用于Unity Objects的各个方面&#xff0c;并可以通过E…

HCIP学习-IPv6

目录 前置学习内容 IPv6解决的一些IPv4的缺陷 无限的地址 层次化的地址结构 即插即用 简化报文头部 IPv4和IPv6报头比较 端到端的网络罗完整性 安全性增强 挣钱QoS特性 IPv6地址介绍 格式 首选格式 压缩格式 内嵌IPv4地址格式的IPv6地址格式 IPv6的网络前缀和接…

MySQL——数据的增删改

2023.9.12 本章开始学习DML (数据操纵语言) 语言。相关学习笔记如下&#xff1a; #DML语言 /* 数据操作语言&#xff1a; 插入&#xff1a;insert 修改&#xff1a;update 删除&#xff1a;delete */#一、插入语句 #方式一&#xff1a;经典的插入 /* 语法&#xff1a; insert …

python协程学习

import asyncio import time import csv import queue import aiosqlite import timeconn None # 定义一个队列&#xff0c;用于传递数据 data_queue queue.Queue()# 启动写文件 # def callback(): # await # print(f"执行结果:{future.result()}") async…

后端入门教程:从零开始学习后端开发

1. 编程基础 首先&#xff0c;作为一名后端开发者&#xff0c;你需要掌握至少一门编程语言。Python是一个很好的选择&#xff0c;因为它易于学习且功能强大。让我们从一个简单的示例开始&#xff0c;在控制台输出 "Hello, World!"。 2. 学习Web基础 了解Web开发基…