搜索
首页后端开发php教程一道简单的算法题

比如
a物品价值1,
b物品价值2,
c物品价值8,
d物品价值10

我给出一个数字 30
求可能的组合

要求为所用到的物品最少,可重复

回复内容:

比如
a物品价值1,
b物品价值2,
c物品价值8,
d物品价值10

我给出一个数字 30
求可能的组合

要求为所用到的物品最少,可重复

@Ukyoi_D 同学的分析很对,只是后面提出的BFS求解剪枝方法并非动态规划的常见方法,当然,方法也是正确的。

首先规范一下原题:

有n个物品,每个物品的价值为v[i],且 v[0]

算法是动态规划,证明过程Ukyoi_D 同学已经证明了。这里给出状态转移方程。

状态转移方程为:

<code> f[i][w] = min{
   f[i-1][w] if f[i-1][w] exist 
   , 
   f[i-1][w-v[i]] +1 if f[i-1][w-v[i]]exist
  }</code>

其中f[i][w]代表在前i个物品中凑足w价值的物品需要的物品数量,f[i][w]存在表示前i个物品可以组合出w价值的物品,不存在表示不能组合出w价值的物品。

对于 f[i][w] 实际上存在3种可能,1. 选i物品,则需要物品数等于f[i-1][w-v[i]]+1。2. 不选i物品,则需要物品等于f[i-1][w], 3. f[i-1][w-v[i]]f[i-1][w]都不存在,则f[i][w]必然也不存在。

三种情况中选出最小的,或者不存在。

具体实现上只需要一个2维数组即可,不存在的状态用无穷大表示。

这个应该不涉及到算法啥的,要求所用到的物品最少,那就是尽量使用价值大的物品,
用30从d-》a取整,取余,整数就是当前的要用多少个,余数是剩下的,继续往下运算,直到分配完为止

//===============================
先看一下问题:
1、没说30一定要分配完
2、要求物品最少

所以我觉得我给出解法没什么问题,勉强算贪心算法

大家都说是背包问题,我认为不是
1、背包问题是两个维度考虑,重量和价值而这里只有价值
2、背包问题要求最后价值最大化,这里仅要求物品最少

问题描述:

<code>有m个物品, 价值从小到大 为V1 - Vm 
f(i) 为 "价值i且物品最少的组合"的 物品数
求解 f(n)
</code>

初始化

<code>f(0) = 0
f(i) = +∞,1 </code>

求解:

<code>从 1到 n
f(i) = min(f(i - Vx) {1 = Vx}) + 1 
如果不满足 1 = Vx, f(i) = +∞</code>

所用到的物品最少,可重复

那就是三个 d,刚好 30。

还有比这个更少的吗?

另外,这个根本不是背包问题啊,背包问题不仅有价值限制,还有重量限制的啊。

楼上说这个问题“不涉及算法”显然是不对的……这是一个多么经典的动态规划入门题……
贪心解对30这个特定的值是可行,但比如对16就是不可行的,16的最优解显然是拿两个8,而不是贪心法的先拿10再拿三个2。

原题中要用最少的物品凑够30,假如我先随便拿一个,比如我拿了一个2,接下来我还要拿28才能满足“凑够30”的要求,于是就出现了一个子问题:如何用最少物品凑够28。同理如果我拿了一个10,子问题就是我需要凑够20。

这个题目的关键在于所谓的“最优子结构”:当且仅当子问题的解是最优的,母问题的解才是最优的。举例而言,拿第一个东西我可以有1、2、8、10四种选择,于是这个问题变成四个子问题,假如我已经知道了四个子问题中的某一个的答案是这四个子问题中所拿东西最少的,就比如说需要拿k个吧,那么加上刚才拿的第一个,那么原问题就最少需要k+1个。当然实际上我们还不知道这4个子问题究竟哪个的解是最优的,所以就采用递归的方法,对四个子问题分别再求子问题,找到所有的二级子问题。在所有二级子问题之中,假如存在一个最优解,比如需要拿m个,那么同理,一级子问题的解就是至少要拿m+1个,而母问题的解就是要拿m+2个。最优子结构是动态规划实现的基础。

具体算法的实现请看 bluedream 君的解答,那个是动态规划的标准方法。我下面给出的解答用了递归,计算机需要更多资源来处理函数调用,而且通常也更慢。


广度优先搜索解法:

刚才的推理建立在“假如知道子问题的解”,当然二级子问题的解实际还是不知道,那就继续往下找。就这么像剥洋葱一样层层剥开(黑话叫“广度优先搜索”)。(这里有个小trick:先拿8再拿2和先拿2再拿8是一样的,优化掉后可以省一小半资源)

求到若干层之后,比如有一条路线最新的子问题要求拿满8,此时显然拿一个8就是这个子问题的最优解。那么别的路线都可以不用算了,因为别的路线最少也得再拿一个才满足要求。顺着这个找到的最优解一层一层退回去,就找到了整个问题的最优解(之一)。当然也可能某个路线最后发现无解(本题不可能,因为能拿1,总能用1凑够任何一个正整数,但如果原题目要求拿30块零5毛那么显然无解),那就舍弃这个路线,继续找其它的路线,直到找到解。或者遍历所有可能的情况发现无解。

当然上面是“广度优先”的方法,也可以用“深度优先”的方法,就是说像走迷宫一样,先顺着其中一个子问题(以及这个子问题的子问题)一条道走到黑,如果是死胡同再退回来,换一条路,这个时候找到的第一个解未必是最优解,但是有了这个解的参照,所有比这个解还要深的胡同就可以都标记成死路,不用再钻了,因为最优解的步骤比刚发现的那个肯定只少不多。如果之后又发现了更优解,就以这个更优的解作新的参照……这样钻过所有的路之后,要么一个最优解也没发现,要么最新发现的解就是最优解。

首先,所有手算结果的回答显然是错误的。
题主拿这个只是举个例子,实际情况当然物品价值和要求解的价值都是程序输入。
不可能手算出一个结果就说这题没意义了。

其次,上面@ekea0407 所说的贪心解法是错误的。
假设物品重量按题目说是1,2,8,10,要求解的重量是16,这个时候按贪心算法:
16=10+6=10+2+4=10+2+2+2,变成4件物品。
其实16=8+8,只需要2件物品。

最后,这个问题和背包问题还是有点区别吧,只是都用动态规划求解,求解过程也类似。
@brayden 的解法就行。
用暴力搜索遍历解空间的解法是很差的,因为暴力搜索没有把已经找到过的局部解存储下来,会多次求解同一问题。
这个题还是用动态规划合适。

感觉楼主这个问题几乎不涉及算法. 先用3个10,直接30了.如果有不用10的话,直接就比3个多了,问题就没有进行下去的必要了.

背包问题
动态规划

这是背包问题,上面一位同学说这个不涉及到算法,其实是不科学的。题主可以网上搜索一下楼教主的‘背包九讲’,这个问题就迎刃而解了。

声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
您什么时候使用特质与PHP中的抽象类或接口?您什么时候使用特质与PHP中的抽象类或接口?Apr 10, 2025 am 09:39 AM

在PHP中,trait适用于需要方法复用但不适合使用继承的情况。1)trait允许在类中复用方法,避免多重继承复杂性。2)使用trait时需注意方法冲突,可通过insteadof和as关键字解决。3)应避免过度使用trait,保持其单一职责,以优化性能和提高代码可维护性。

什么是依赖性注入容器(DIC),为什么在PHP中使用一个?什么是依赖性注入容器(DIC),为什么在PHP中使用一个?Apr 10, 2025 am 09:38 AM

依赖注入容器(DIC)是一种管理和提供对象依赖关系的工具,用于PHP项目中。DIC的主要好处包括:1.解耦,使组件独立,代码易维护和测试;2.灵活性,易替换或修改依赖关系;3.可测试性,方便注入mock对象进行单元测试。

与常规PHP阵列相比,解释SPL SplfixedArray及其性能特征。与常规PHP阵列相比,解释SPL SplfixedArray及其性能特征。Apr 10, 2025 am 09:37 AM

SplFixedArray在PHP中是一种固定大小的数组,适用于需要高性能和低内存使用量的场景。1)它在创建时需指定大小,避免动态调整带来的开销。2)基于C语言数组,直接操作内存,访问速度快。3)适合大规模数据处理和内存敏感环境,但需谨慎使用,因其大小固定。

PHP如何安全地上载文件?PHP如何安全地上载文件?Apr 10, 2025 am 09:37 AM

PHP通过$\_FILES变量处理文件上传,确保安全性的方法包括:1.检查上传错误,2.验证文件类型和大小,3.防止文件覆盖,4.移动文件到永久存储位置。

什么是无效的合并操作员(??)和无效分配运算符(?? =)?什么是无效的合并操作员(??)和无效分配运算符(?? =)?Apr 10, 2025 am 09:33 AM

JavaScript中处理空值可以使用NullCoalescingOperator(??)和NullCoalescingAssignmentOperator(??=)。1.??返回第一个非null或非undefined的操作数。2.??=将变量赋值为右操作数的值,但前提是该变量为null或undefined。这些操作符简化了代码逻辑,提高了可读性和性能。

什么是内容安全策略(CSP)标头,为什么重要?什么是内容安全策略(CSP)标头,为什么重要?Apr 09, 2025 am 12:10 AM

CSP重要因为它能防范XSS攻击和限制资源加载,提升网站安全性。1.CSP是HTTP响应头的一部分,通过严格策略限制恶意行为。2.基本用法是只允许从同源加载资源。3.高级用法可设置更细粒度的策略,如允许特定域名加载脚本和样式。4.使用Content-Security-Policy-Report-Only头部可调试和优化CSP策略。

什么是HTTP请求方法(获取,发布,放置,删除等),何时应该使用?什么是HTTP请求方法(获取,发布,放置,删除等),何时应该使用?Apr 09, 2025 am 12:09 AM

HTTP请求方法包括GET、POST、PUT和DELETE,分别用于获取、提交、更新和删除资源。1.GET方法用于获取资源,适用于读取操作。2.POST方法用于提交数据,常用于创建新资源。3.PUT方法用于更新资源,适用于完整更新。4.DELETE方法用于删除资源,适用于删除操作。

什么是HTTP,为什么对Web应用程序至关重要?什么是HTTP,为什么对Web应用程序至关重要?Apr 09, 2025 am 12:08 AM

HTTPS是一种在HTTP基础上增加安全层的协议,主要通过加密数据保护用户隐私和数据安全。其工作原理包括TLS握手、证书验证和加密通信。实现HTTPS时需注意证书管理、性能影响和混合内容问题。

See all articles

热AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover

AI Clothes Remover

用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool

Undress AI Tool

免费脱衣服图片

Clothoff.io

Clothoff.io

AI脱衣机

AI Hentai Generator

AI Hentai Generator

免费生成ai无尽的。

热门文章

R.E.P.O.能量晶体解释及其做什么(黄色晶体)
3 周前By尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.最佳图形设置
3 周前By尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.如果您听不到任何人,如何修复音频
3 周前By尊渡假赌尊渡假赌尊渡假赌
WWE 2K25:如何解锁Myrise中的所有内容
3 周前By尊渡假赌尊渡假赌尊渡假赌

热工具

PhpStorm Mac 版本

PhpStorm Mac 版本

最新(2018.2.1 )专业的PHP集成开发工具

Dreamweaver Mac版

Dreamweaver Mac版

视觉化网页开发工具

禅工作室 13.0.1

禅工作室 13.0.1

功能强大的PHP集成开发环境

螳螂BT

螳螂BT

Mantis是一个易于部署的基于Web的缺陷跟踪工具,用于帮助产品缺陷跟踪。它需要PHP、MySQL和一个Web服务器。请查看我们的演示和托管服务。

SublimeText3汉化版

SublimeText3汉化版

中文版,非常好用