【题目】给定数组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 : 换钱的方法数的更多相关文章

  1. 算法进阶面试题07——求子数组的最大异或和(前缀树)、换钱的方法数(递归改dp最全套路解说)、纸牌博弈、机器人行走问题

    主要讲第五课的内容前缀树应用和第六课内容暴力递归改动态规划的最全步骤 第一题 给定一个数组,求子数组的最大异或和. 一个数组的异或和为,数组中所有的数异或起来的结果. 简单的前缀树应用 暴力方法: 先 ...

  2. [DP]换钱的方法数

    题目三 给定数组arr, arr中所有的值都为整数且不重复.每个值代表一种面值的货币,每种面值的货币可以使用任意张,在给定一个整数aim代表要找的钱数,求换钱有多少种方法. 解法一 --暴力递归 用0 ...

  3. [程序员代码面试指南]递归和动态规划-换钱的方法数(DP,完全背包)

    题目描述 给定arr,arr中所有的值都为正数且不重复.每个值代表一种面值的货币,每种面值的货币可以使用任意张,再给定一个整数aim,求组成aim的方法数. 解题思路 完全背包 和"求换钱的 ...

  4. 八大排序算法的 Python 实现

    转载: 八大排序算法的 Python 实现 本文用Python实现了插入排序.希尔排序.冒泡排序.快速排序.直接选择排序.堆排序.归并排序.基数排序. 1.插入排序 描述 插入排序的基本操作就是将一个 ...

  5. Python练习题 003:完全平方数

    [Python练习题 003]一个整数,它加上100后是一个完全平方数,再加上168又是一个完全平方数,请问该数是多少? --------------------------------------- ...

  6. 数据关联分析 association analysis (Aprior算法,python代码)

    1基本概念 购物篮事务(market basket transaction),如下表,表中每一行对应一个事务,包含唯一标识TID,和购买的商品集合.本文介绍一种成为关联分析(association a ...

  7. 机器学习算法与Python实践之(四)支持向量机(SVM)实现

    机器学习算法与Python实践之(四)支持向量机(SVM)实现 机器学习算法与Python实践之(四)支持向量机(SVM)实现 zouxy09@qq.com http://blog.csdn.net/ ...

  8. 机器学习算法与Python实践之(三)支持向量机(SVM)进阶

    机器学习算法与Python实践之(三)支持向量机(SVM)进阶 机器学习算法与Python实践之(三)支持向量机(SVM)进阶 zouxy09@qq.com http://blog.csdn.net/ ...

  9. 机器学习算法与Python实践之(二)支持向量机(SVM)初级

    机器学习算法与Python实践之(二)支持向量机(SVM)初级 机器学习算法与Python实践之(二)支持向量机(SVM)初级 zouxy09@qq.com http://blog.csdn.net/ ...

随机推荐

  1. R-shiny服务器安装及配置

    R的shiny 服务器管理-入门http://www.bio-info-trainee.com/1683.html sudo systemctl start shiny-serversudo syst ...

  2. 深入理解CSS选择器优先级

    题外话 今天把 <CSS REFACTORING>(中文名叫<CSS重构:样式表性能调优>)电子书粗略的浏览了一遍,这本书很薄,150页左右,首先是介绍了什么是重构并举了两个简 ...

  3. install rust

    Step 1. Trial 1 Download rustup-init.exe exec rustup-init.exe SW hangs 2. Trial 2 install rust-1.33. ...

  4. python-变量、if else语句 、for循环、while循环(4月26号)

    变量: 五.注意:python是可执行程序 在linux写python第一行必须写#!/usr/bin/env python(声明解释器在windows中写python第一行需要写# -*- codi ...

  5. Windbg 内核态调试用户态程序然后下断点正确触发方法(亲自实现发现有效)

    先开启真机内核态kernel调试 !process 0 0 svchost.exe 找到进程cid的地址 然后进入 .process /p  fffffa8032be2870 然后 .process ...

  6. centos7配置iscsi

    什么是ISCSI iscsi--internet small computer system interface互联小型计算机系统接口,将数据包封装在TCP/IP协议中传输,使用普通网线和网络设备即可 ...

  7. 咱家自己的vim配置

    " 四个空格设置 set tabstop=4 set softtabstop=4 set shiftwidth=4 set autoindent set smartindent set ex ...

  8. [ZZ] 多领域视觉数据的转换、关联与自适应学习

    哈工大左旺孟教授:多领域视觉数据的转换.关联与自适应学习 http://blog.sciencenet.cn/home.php?mod=space&uid=3291369&do=blo ...

  9. 支持向量机(Support Vector Machine,SVM)

    SVM: 1. 线性与非线性 核函数: 2. 与神经网络关系 置信区间结构: 3. 训练方法: 4.SVM light,LS-SVM: 5. VC维 u-SVC 与 c-SVC 区别? 除参数不同外, ...

  10. AR图像识别 AR识别图像 AR摄像头识别 外包开发 AR识别应用开发就找北京动点软件

    当绝大多数手机厂商还在追求后置双摄拍照的时候,已经有人开始潜心研究AR手机了.刚刚结束的美国消费电子展上,华硕发布了全新的ZenFone AR手机,配备5.7英寸2K屏.骁龙821处理器.8GB内存, ...