算法之Python实现 - 003 : 换钱的方法数
【题目】给定数组arr,arr中所有的值都为正数且不重复。每个值代表一种面值的货币,每种面值的货币可以使用任意张,再给定一个整数aim代表要找的钱数,求组成aim的方法数。
【代码1】递归
import numpy as np def changemeans(arr,aim):
if len(arr)<0:
print("No coin provided for change!")
arr.sort()
arr.reverse()
m = process(arr,0,aim)
print('There are ',m,' ways!') def process(arr,idx,aim):
res = 0
i = 0
if aim == 0:
res = 1
else :
if idx == len(arr):
res = 0
else :
while arr[idx]*i <= aim:
res += process(arr,idx+1,aim - arr[idx]*i)
i += 1
return res # ===CALL === #
a = [5,10,25,1]
tar = 1000
changemeans(a,tar)
【代码2】改进递归(递归加入记忆搜索):时间复杂度O(N * aim2)
【原理】:例如按照题目中的a = [5,10,25,1],使用a[0]和a[1],利用[25,1]组成剩余的980元的可能性就是一种重复递归,假设利用[25,1]组成剩余的980元需要5秒钟,那么【代码1】需要搜索5*0+10*2,5*2+10*1,5*5 三次递归,【代码2】额外耗用了O((N+1)*(aim+1))的空间,但是只要三次寻址即可。
import numpy as np def changemeans(arr,aim):
if len(arr)<0:
print("No coin provided for change!")
arr.sort()
arr.reverse()
map = np.zeros((len(arr)+1,aim+1))
m = process(arr,0,aim,map)
print('There are ',m,' ways!') def process(arr,idx,aim,map):
res = 0
i = 0
if aim == 0:
res = 1
else :
if idx == len(arr):
res = 0
else :
while arr[idx]*i <= aim:
mapval = map[idx+1][aim- arr[idx]*i]
if mapval != 0:
if mapval == -1: mapval = 0
res += mapval
else:
res += process(arr,idx+1,aim - arr[idx]*i,map)
i += 1
if res == 0:
map[idx][aim] = -1
else :
map[idx][aim] = res
#print(':',int(map[idx][aim]),res)
return res # ===CALL === #
a = [5,10,25,1]
tar = 1000
changemeans(a,tar)
【代码3】:时间复杂度O(N * aim2)
import numpy as np def changemeans(arr,aim):
n = len(arr)
if n<=0:
print('No coin provided for exchange.')
j = 0
dp = np.zeros((n,aim+1)) for i in range(0,n):
dp[i][0] = 1 while j*arr[0]<= aim:
dp[0][j*arr[0]] = 1
j += 1 for i in range(1,n):
for j in range(1,aim+1):
num = 0
k = 0
while j-arr[i]*k >= 0:
num += dp[i-1][j-arr[i]*k]
k += 1
dp[i][j] = num print(dp[n-1][aim]) # ===CALL === #
a = [5,10,25,1]
tar = 1000
changemeans(a,tar)
【代码4】:
另外实际上算arr[0..i-1]的组成剩下的方法,只会从最少的那个钱币为下标的位置开始,因此可以改为:
import numpy as np def changemeans(arr,aim):
n = len(arr)
if n<=0:
print('No coin provided for exchange.')
j = 0
dp = np.zeros((n,aim+1)) for i in range(0,n):
dp[i][0] = 1 while j*arr[0]<= aim:
dp[0][j*arr[0]] = 1
j += 1 for i in range(1,n):
for j in range(min(arr)-1,aim+1):
num = 0
k = 0
while j-arr[i]*k >= 0:
num += dp[i-1][j-arr[i]*k]
k += 1
dp[i][j] = num print(dp[n-1][aim]) # ===CALL === #
a = [5,10,25,2]
tar = 1000
changemeans(a,tar)
算法之Python实现 - 003 : 换钱的方法数的更多相关文章
- 算法进阶面试题07——求子数组的最大异或和(前缀树)、换钱的方法数(递归改dp最全套路解说)、纸牌博弈、机器人行走问题
主要讲第五课的内容前缀树应用和第六课内容暴力递归改动态规划的最全步骤 第一题 给定一个数组,求子数组的最大异或和. 一个数组的异或和为,数组中所有的数异或起来的结果. 简单的前缀树应用 暴力方法: 先 ...
- [DP]换钱的方法数
题目三 给定数组arr, arr中所有的值都为整数且不重复.每个值代表一种面值的货币,每种面值的货币可以使用任意张,在给定一个整数aim代表要找的钱数,求换钱有多少种方法. 解法一 --暴力递归 用0 ...
- [程序员代码面试指南]递归和动态规划-换钱的方法数(DP,完全背包)
题目描述 给定arr,arr中所有的值都为正数且不重复.每个值代表一种面值的货币,每种面值的货币可以使用任意张,再给定一个整数aim,求组成aim的方法数. 解题思路 完全背包 和"求换钱的 ...
- 八大排序算法的 Python 实现
转载: 八大排序算法的 Python 实现 本文用Python实现了插入排序.希尔排序.冒泡排序.快速排序.直接选择排序.堆排序.归并排序.基数排序. 1.插入排序 描述 插入排序的基本操作就是将一个 ...
- Python练习题 003:完全平方数
[Python练习题 003]一个整数,它加上100后是一个完全平方数,再加上168又是一个完全平方数,请问该数是多少? --------------------------------------- ...
- 数据关联分析 association analysis (Aprior算法,python代码)
1基本概念 购物篮事务(market basket transaction),如下表,表中每一行对应一个事务,包含唯一标识TID,和购买的商品集合.本文介绍一种成为关联分析(association a ...
- 机器学习算法与Python实践之(四)支持向量机(SVM)实现
机器学习算法与Python实践之(四)支持向量机(SVM)实现 机器学习算法与Python实践之(四)支持向量机(SVM)实现 zouxy09@qq.com http://blog.csdn.net/ ...
- 机器学习算法与Python实践之(三)支持向量机(SVM)进阶
机器学习算法与Python实践之(三)支持向量机(SVM)进阶 机器学习算法与Python实践之(三)支持向量机(SVM)进阶 zouxy09@qq.com http://blog.csdn.net/ ...
- 机器学习算法与Python实践之(二)支持向量机(SVM)初级
机器学习算法与Python实践之(二)支持向量机(SVM)初级 机器学习算法与Python实践之(二)支持向量机(SVM)初级 zouxy09@qq.com http://blog.csdn.net/ ...
随机推荐
- HI3518EV200+AR0130开发板烧录uboot、kernel、rootfs及其参数配置
分区名 分区大小 起始地址 截至地址bootloader:1M 0x00000000 0x00100000kernel: 3M 0x00100000 0x00400000rootfs: 12M 0x0 ...
- 学习Python3 试了一下百度OCR和腾讯OCR
因为有个小功能,需要用一下OCR,所以先找了2家,百度和腾讯,如何开通,如何创建应用获得key等不作说明了 百度的比较简单,引用一个AipOcr全部搞定,代码如下: from aip import A ...
- python随机数学习笔记
#coding:utf-8 import random # random.randint(1,10)产生1,10的随机整数 for i in range(1,5): ranint = random.r ...
- pycharm 激活码及使用方式
https://www.cnblogs.com/pupilheart/p/9734124.html https://www.cnblogs.com/pupilheart/p/9084127.html ...
- Python高级变量类型
1.列表基础操作 2.列表循环遍历 3.元组 4.字典基础操作 5.字典循环遍历 6.字符串基础操作 7.字符串切片 8.公共方法
- 自定义页面微信、微博、QQ分享效果
几行简单的分享代码既可以实现,先看下效果: 第一步:页面因为结构代码 <div id="freebtn"> <ul> <li class=" ...
- 更改/var/log/messages默认权限
问题描述: 操作系统:redhat 6.5 因为开发人员和运维人员平时在应用出现bug时,会需要看/var/log/message日志,但是默认权限为600,因此除了root用户都无法读取,现需要把每 ...
- springMvc接收ajax数组参数,以及jquery复选框选中、反选、全选、全不选
一.复选框选中.反选.全选.全不选 html代码: <input type='checkbox' name='menuCheckBox' value='10' >苹果 <input ...
- 如何启用小米手机5c的ROOT权限
小米手机5c怎么样开通了root超级权限?大家都知道,android设备有root超级权限,一旦手机开通了root相关权限,能够实现更强大的功能,举个例子大家部门的营销部门的同事,使用某些营销应用都需 ...
- windows 下借助7zip实现命令行解压缩
windows 下借助7zip实现命令行解压缩 64位电脑下载 https://www.7-zip.org/a/7z1805-x64.exe 安装 安装目录下所有文件如下: 在命令行下只需要用到 7z ...