目录
回复内容:
首页 后端开发 php教程 一道简单的算法题

一道简单的算法题

Jun 06, 2016 pm 08:26 PM
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

热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无尽的。

热工具

记事本++7.3.1

记事本++7.3.1

好用且免费的代码编辑器

SublimeText3汉化版

SublimeText3汉化版

中文版,非常好用

禅工作室 13.0.1

禅工作室 13.0.1

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

Dreamweaver CS6

Dreamweaver CS6

视觉化网页开发工具

SublimeText3 Mac版

SublimeText3 Mac版

神级代码编辑软件(SublimeText3)

适用于 Ubuntu 和 Debian 的 PHP 8.4 安装和升级指南 适用于 Ubuntu 和 Debian 的 PHP 8.4 安装和升级指南 Dec 24, 2024 pm 04:42 PM

PHP 8.4 带来了多项新功能、安全性改进和性能改进,同时弃用和删除了大量功能。 本指南介绍了如何在 Ubuntu、Debian 或其衍生版本上安装 PHP 8.4 或升级到 PHP 8.4

如何设置 Visual Studio Code (VS Code) 进行 PHP 开发 如何设置 Visual Studio Code (VS Code) 进行 PHP 开发 Dec 20, 2024 am 11:31 AM

Visual Studio Code,也称为 VS Code,是一个免费的源代码编辑器 - 或集成开发环境 (IDE) - 可用于所有主要操作系统。 VS Code 拥有针对多种编程语言的大量扩展,可以轻松编写

我后悔之前不知道的 7 个 PHP 函数 我后悔之前不知道的 7 个 PHP 函数 Nov 13, 2024 am 09:42 AM

如果您是一位经验丰富的 PHP 开发人员,您可能会感觉您已经在那里并且已经完成了。您已经开发了大量的应用程序,调试了数百万行代码,并调整了一堆脚本来实现操作

您如何在PHP中解析和处理HTML/XML? 您如何在PHP中解析和处理HTML/XML? Feb 07, 2025 am 11:57 AM

本教程演示了如何使用PHP有效地处理XML文档。 XML(可扩展的标记语言)是一种用于人类可读性和机器解析的多功能文本标记语言。它通常用于数据存储

在PHP API中说明JSON Web令牌(JWT)及其用例。 在PHP API中说明JSON Web令牌(JWT)及其用例。 Apr 05, 2025 am 12:04 AM

JWT是一种基于JSON的开放标准,用于在各方之间安全地传输信息,主要用于身份验证和信息交换。1.JWT由Header、Payload和Signature三部分组成。2.JWT的工作原理包括生成JWT、验证JWT和解析Payload三个步骤。3.在PHP中使用JWT进行身份验证时,可以生成和验证JWT,并在高级用法中包含用户角色和权限信息。4.常见错误包括签名验证失败、令牌过期和Payload过大,调试技巧包括使用调试工具和日志记录。5.性能优化和最佳实践包括使用合适的签名算法、合理设置有效期、

php程序在字符串中计数元音 php程序在字符串中计数元音 Feb 07, 2025 pm 12:12 PM

字符串是由字符组成的序列,包括字母、数字和符号。本教程将学习如何使用不同的方法在PHP中计算给定字符串中元音的数量。英语中的元音是a、e、i、o、u,它们可以是大写或小写。 什么是元音? 元音是代表特定语音的字母字符。英语中共有五个元音,包括大写和小写: a, e, i, o, u 示例 1 输入:字符串 = "Tutorialspoint" 输出:6 解释 字符串 "Tutorialspoint" 中的元音是 u、o、i、a、o、i。总共有 6 个元

解释PHP中的晚期静态绑定(静态::)。 解释PHP中的晚期静态绑定(静态::)。 Apr 03, 2025 am 12:04 AM

静态绑定(static::)在PHP中实现晚期静态绑定(LSB),允许在静态上下文中引用调用类而非定义类。1)解析过程在运行时进行,2)在继承关系中向上查找调用类,3)可能带来性能开销。

什么是PHP魔术方法(__ -construct,__destruct,__call,__get,__ set等)并提供用例? 什么是PHP魔术方法(__ -construct,__destruct,__call,__get,__ set等)并提供用例? Apr 03, 2025 am 12:03 AM

PHP的魔法方法有哪些?PHP的魔法方法包括:1.\_\_construct,用于初始化对象;2.\_\_destruct,用于清理资源;3.\_\_call,处理不存在的方法调用;4.\_\_get,实现动态属性访问;5.\_\_set,实现动态属性设置。这些方法在特定情况下自动调用,提升代码的灵活性和效率。

See all articles