Python非单向递归函数如何返回全部结果


Posted in Python onDecember 18, 2020

递归( recursion)是一种神奇的编程技巧,可以大幅简化代码,使之看起来更加简洁。然而递归设计却非常抽象,不容易掌握。通常,我们都是自上而下的思考问题, 递归则是自下而上的解决问题——这就是递归看起来不够直观的原因。

和递归相关的概念里,线性递归/非线性递归、单向递归/非单向递归,是非常重要的,要想掌握递归技术,就必须要深入理解。关于递归的基本概念,有兴趣的读者,可以参考我的博客《Python 递归算法指归》。今天,仅就背包问题谈非单向递归函数如何返回全部结果。

背包问题的背后,是世界七大数学难题之一,多项式复杂程度的非确定性问题。作为程序员,可以将该问题大致上理解为组合优化的问题。背包问题通常被这样描述:给定一组物品,每种物品都有自己的重量和价格,在限定的总重量内,如何选择,才能使得物品的总价格最高。如果加上不同的限制和条件,背包问题可以衍生出很多变种。比如,下面这道题看起来和背包问题相去甚远,实质上仍然是一个典型的背包问题。

在一款英雄对战游戏中,玩家拥有m件装备和n位英雄,他可以给每一位英雄分配0件或多件装备,而不同的英雄拥有不同数目的装备时将获得不同的攻击力。玩家如何分配这m件装备,可以使得n个英雄获得的攻击力的和最大?以玩家拥有5件装备和3位英雄为例,下表共有3行6列,对应着3位英雄分别拥有从0到5件装备时的攻击力。

0件 1件 2件 3件 4件 5件
英雄1 0 1 3 5 7 9
英雄2 0 1 1 3 3 7
英雄3 0 3 4 5 6 7

即使不熟悉背包问题,也不难找到解题思路:

  • 找出所有可能的装备分配方案
  • 计算每一个方案的攻击值
  • 选择攻击值最大的分配方案

1. 找出所有可能的装备分配方案

找出将m件装备分配给n位英雄的所有方案是解决问题的核心。这里,循环嵌套是行不通的,因为嵌套层数是输入变量。递归是我想到的可行的方法。

>>> def bag(m, n, series=list()):
    if n == 1:
      for i in range(m+1):
        print(series+[i])
    else:
      for i in range(m+1):
        bag(m-i, n-1, series+[i])
  
>>> bag(3,2) # 将3件装备分配给2位英雄的全部方案
[0, 0]
[0, 1]
[0, 2]
[0, 3]
[1, 0]
[1, 1]
[1, 2]
[2, 0]
[2, 1]
[3, 0]

递归函数bag,打印出了将3件装备分配给2位英雄的全部方案。显然,这不是一个单向递归,因为在同一级有多次递归调用,这意味着递归过程有多次从递归出口走出。对于非单向递归,是不能使用return返回结果的。那么,如何让递归函数返回全部方案呢?请看下面的例子。

>>> def bag(m, n, result, series=list()):
 if n == 1:
 for i in range(m+1):
  result.append(series+[i])
  #print(result[-1])
 else:
 for i in range(m+1):
  bag(m-i, n-1, result, series+[i])

  
>>> result = list()
>>> bag(5, 3, result) # 将5件装备分配给3位英雄,共有56种分配方案
>>> len(result)
56
>>> result
[[0, 0, 0], [0, 0, 1], [0, 0, 2], [0, 0, 3], [0, 0, 4], [0, 0, 5], 
[0, 1, 0], [0, 1, 1], [0, 1, 2], [0, 1, 3], [0, 1, 4], [0, 2, 0], 
[0, 2, 1], [0, 2, 2], [0, 2, 3], [0, 3, 0], [0, 3, 1], [0, 3, 2], 
[0, 4, 0], [0, 4, 1], [0, 5, 0], [1, 0, 0], [1, 0, 1], [1, 0, 2], 
[1, 0, 3], [1, 0, 4], [1, 1, 0], [1, 1, 1], [1, 1, 2], [1, 1, 3], 
[1, 2, 0], [1, 2, 1], [1, 2, 2], [1, 3, 0], [1, 3, 1], [1, 4, 0], 
[2, 0, 0], [2, 0, 1], [2, 0, 2], [2, 0, 3], [2, 1, 0], [2, 1, 1], 
[2, 1, 2], [2, 2, 0], [2, 2, 1], [2, 3, 0], [3, 0, 0], [3, 0, 1], 
[3, 0, 2], [3, 1, 0], [3, 1, 1], [3, 2, 0], [4, 0, 0], [4, 0, 1], 
[4, 1, 0], [5, 0, 0]]

上面的代码中,在调用递归函数之前,先创建一个全局的列表对象result,并作为参数传递给递归函数。递归调用结束后,全部的装备分配方案就保存在列表对象result中。

2. 计算每一个方案的攻击值

遍历56种分配方案,计算每一种方案的攻击力之和,保存到一个新的列表v中。p为3位英雄分别拥有从0到5件装备时的攻击力。

>>> p = [
 [0,1,3,5,7,9],
 [0,1,1,3,3,7],
 [0,3,4,5,6,7]
]
>>> v = list()
>>> for item in result:
    v.append(p[0][item[0]] + p[1][item[1]] + p[2][item[2]])
 
>>> v
[0, 3, 4, 5, 6, 7, 1, 4, 5, 6, 7, 1, 4, 5, 6, 3, 6, 7, 3,
 6, 7, 1, 4, 5, 6, 7, 2, 5, 6, 7, 2, 5, 6, 4, 7, 4, 3, 6, 
 7, 8, 4, 7, 8, 4, 7, 6, 5, 8, 9, 6, 9, 6, 7, 10, 8, 9]

3. 选择攻击值最大的分配方案

找出v列表最大值的序号,进而得到攻击力最大的装备分配方案。

>>> max(v)
10
>>> result[v.index(max(v))] 
[4, 0, 1]

最佳分配方案是第1位英雄持有4件装备,第2位英雄没有装备,第3位英雄持有1件装备,此时3位英雄的攻击力之和为最大,其值为10。

到此这篇关于Python非单向递归函数如何返回全部结果的文章就介绍到这了,更多相关Python非单向递归返回 内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

Python 相关文章推荐
flask中使用SQLAlchemy进行辅助开发的代码
Feb 10 Python
python通过正则查找微博@(at)用户的方法
Mar 13 Python
Python的Socket编程过程中实现UDP端口复用的实例分享
Mar 19 Python
Python 转义字符详细介绍
Mar 21 Python
Python反射用法实例简析
Dec 22 Python
python中字符串数组逆序排列方法总结
Jun 23 Python
Django REST Framework序列化外键获取外键的值方法
Jul 26 Python
简单了解Java Netty Reactor三种线程模型
Apr 26 Python
Python中的__init__作用是什么
Jun 09 Python
python语言中有算法吗
Jun 16 Python
python中查看.db文件中表格的名字及表格中的字段操作
Jul 07 Python
Python利器openpyxl之操作excel表格
Apr 17 Python
python复合条件下的字典排序
Dec 18 #Python
python 监控服务器是否有人远程登录(详细思路+代码)
Dec 18 #Python
细说NumPy数组的四种乘法的使用
Dec 18 #Python
如何通过python检查文件是否被占用
Dec 18 #Python
python 实现端口扫描工具
Dec 18 #Python
Python-split()函数实例用法讲解
Dec 18 #Python
Python+Opencv实现把图片、视频互转的示例
Dec 17 #Python
You might like
php adodb连接带密码access数据库实例,测试成功
2008/05/14 PHP
PHP 类商品秒杀计时实现代码
2010/05/05 PHP
对象失去焦点时自己动提交数据的实现代码
2012/11/06 PHP
PHP禁止个别IP访问网站
2013/10/30 PHP
ThinkPHP3.1新特性之G方法的使用
2014/06/19 PHP
ThinkPHP的MVC开发机制实例解析
2014/08/23 PHP
ThinkPHP实现二级循环读取的方法
2014/11/03 PHP
详解如何实现Laravel的服务容器的方法示例
2019/04/15 PHP
tp5修改(实现即点即改)
2019/10/18 PHP
使用JavaScript 实现对象 匀速/变速运动的方法
2013/05/08 Javascript
JS实现点击按钮后框架内载入不同网页的方法
2015/05/05 Javascript
jQuery 调用WebService 实例讲解
2016/06/28 Javascript
js 定义对象数组(结合)多维数组方法
2016/07/27 Javascript
JS文件/图片从电脑里面拖拽到浏览器上传文件/图片
2017/03/08 Javascript
vue2.0+SVG实现音乐播放圆形进度条组件
2019/09/21 Javascript
解决node.js含有%百分号时发送get请求时浏览器地址自动编码的问题
2019/11/20 Javascript
如何优雅地取消 JavaScript 异步任务
2020/03/22 Javascript
html-webpack-plugin修改页面的title的方法
2020/06/18 Javascript
[01:31:22]DOTA2-DPC中国联赛定级赛 LBZS vs Magma BO3第二场 1月10日
2021/03/11 DOTA
Python爬虫框架Scrapy安装使用步骤
2014/04/01 Python
Python装饰器的函数式编程详解
2015/02/27 Python
django 框架实现的用户注册、登录、退出功能示例
2019/11/28 Python
Python线程协作threading.Condition实现过程解析
2020/03/12 Python
Python如何读取、写入JSON数据
2020/07/28 Python
Python程序慢的重要原因
2020/09/04 Python
酒店门卫岗位职责
2013/12/29 职场文书
基层党员对照检查材料
2014/09/24 职场文书
同学聚会通知书
2015/04/20 职场文书
2015年法务工作总结范文
2015/05/23 职场文书
高中同学会致辞
2015/08/01 职场文书
小学体育组工作总结
2015/08/13 职场文书
观看安全警示教育片心得体会
2016/01/15 职场文书
筑梦中国心得体会
2016/01/18 职场文书
phpQuery解析HTML乱码问题(补充官网未列出的乱码解决方案)
2021/04/01 PHP
Go语言带缓冲的通道实现
2021/04/26 Golang
CSS三大特性继承性、层叠性和优先级详解
2022/01/18 HTML / CSS