洗礼灵魂,修炼python(77)--全栈项目实战篇(5)—— ATM自动存取机系统
要求:
1.完成常识中的ATM存取款机功能
2.把ATM机故障考虑进去
3.不能直接输入账户名和卡号等等信息,模拟出插银行卡让ATM机自动读取卡信息
4.密码验证超过三次错误即锁定账户
5.操作类型有:存款,取款,查询,转账,打印凭条,退卡,一共六个选项,跟用户输入做出不同的反馈
6.当用户正常进入,列出操作类型,每次操作结束后都要重新选择操作类型,只有退卡操作才会结束
7.操作结束后,数据要同步更新
8.要有日志文件,每个涉及到钱走向的操作都要有日志记录
分析:
同样的没什么可说的,根据生活中的场景来就行
关键的问题:怎么模拟插卡操作,让ATM机自动获取信息,方法有:
1.调用API接口读取卡信息,但基本实现起来很难,除非真的搞来一个和现实中的ATM机一样的读卡设备
2.我想了很久,用一个文本文档,里面存放有卡号,用户名等基本信息,然后导入文本的数据来模拟插卡操作,剩下的操作就简单了
我希望你已经思考过再看的下面:
源代码及相关文件源码下载地址:传送门
需要四个文件:
主程序ATM.py
#!usr/bin/env python
#-*- coding:utf-8 -*-
# author:yangva
# datetime:2017/12/16 0015 13:32
import random,time
atm_fault = random.randint(0,100) #ATM机随机故障系数
if atm_fault == 0:
print('ATM出现故障,暂时不可用,抱歉给您带来不便')
else:
print('ATM播放幻灯片,推销新业务(信用卡办理,VIP会员特权,新卡新功能等等)')
#插入卡操作
try:
#模拟读卡的操作,这里用文本文档代替
card = open(input('欢迎光临,请在下方插入口插入您的银行卡(输入文本文件路径和文件名):'))
print('正在读卡,请稍后。。。')
time.sleep(2)
f = eval(card.read())
username = list(f.keys())[0] #取出账户名
user_id = f[username]['ID'] #取出银行卡号
except (NameError,IOError,FileNotFoundError): #输入的文本不存在则捕获
print('插入卡操作过程有误,请检查是否存在该文件或打开该文件的权限')
else:
f = open('database.txt','r')
database = eval(f.read())
f.close() #这里不用with 语句,为保证数据库文件能立即关闭
print('%s先生,欢迎您!!!'%username[0]) #问候语
if username not in database.keys():
print('您的账户已冻结,请本人携带身份证到银行柜台解冻\n已退卡,请收好您的银行卡')
else:
password = input('请输入您的密码(密码为6位数字),请注意周围环境以及遮挡键盘操作:').strip() #后期可以改为语言提示
count = 0 #用于操作计数
ctrl_flag = False #标志位,用于跳出多层循环
while not ctrl_flag:
if count == 2:
print('您已输入多次错误密码,银行账户已冻结')
with open('log.txt','a') as f2: #写入日志
database['freeze'+username] = database.pop(username)
f1 = open('database.txt','w') #同样的,不用with语句,保证数据文件立即关闭
f1.write(str(database))
f2.write(str(time.strftime("%Y %b %d %X"))+'\n用户 %s 的银行卡已冻结\n'%user_id)
f1.close()
ctrl_flag = True
break
if len(str(password)) == 6: #用于判断输入的密码位数是否正确
if password == database[username]['password']:
database_bk = database #做临时备份数据的操作,方便后面出错时会滚
temp = '' #凭条
while not ctrl_flag:
cont = input('1.取款\n2.存款\n3.转账\n4.查询\n5.打印凭条\n6.退卡\n请选择交易类型(输入对应的序号即可):')
#账户主体操作
if cont == '1': #取款
withdraw_money = int(input('请输入取款金额(最低一百):')) #ATM存取都是100的整数,直接用int类型
if withdraw_money %100 == 0:
print('正在点钞,请稍后。。。')
if withdraw_money > database[username]['balance']:
print('操作失败,您输入的金额大于您的余额总数')
else:
time.sleep(3)
print('已成功出钞,请取走您的钞票 %s元'%withdraw_money)
database[username]['balance'] -= withdraw_money
with open('log.txt','a') as f: #写入日志
f.write(str(time.strftime("%Y %b %d %X"))+'\t卡号为 %s 的用户 %s 取出金额:%d\n'%(user_id,username,withdraw_money))
temp += str(time.strftime("%Y %b %d %X"))+'\t取款金额:%d\n'%withdraw_money #记录凭条
else:
print('输入金额不是100的整数倍,请重新输入')
elif cont == '2': #存款
save_money = int(input('请将钞票叠好,并整齐放入存钞口(输入存款金额):')) #ATM存取款都是100的整数,直接用int类型
print('正在点钞,请稍后。。。')
time.sleep(3)
print('%s 元已成功存入账户 %s\t账户名 %s'%(save_money,user_id,username))
database[username]['balance'] += save_money
with open('log.txt','a') as f: #写入日志
f.write(str(time.strftime("%Y %b %d %X"))+'\t卡号为 %s 的用户 %s 已存入金额 %d\n'%(user_id,username,save_money))
temp += str(time.strftime("%Y %b %d %X"))+'\t存入金额:%d\n'%save_money
elif cont == '3': #转账
transfer_id = input('请输入转账帐户卡号:')
transfer_user = input('请输入帐户名:')
if transfer_id == database[transfer_user]['ID']:
transfer_money = float(input('请输入转账金额:'))
print('正在转账,请稍后。。。')
time.sleep(3)
database[username]['balance'] -= transfer_money
database[transfer_user]['balance'] += transfer_money
with open('log.txt','a') as f: #写入日志
f.write(str(time.strftime("%Y %b %d %X"))+'\t卡号为 %s 的用户 %s 转出金额为 %d 给卡号为 %s 的用户 %s\n'%(user_id,username,transfer_money,transfer_id,transfer_user))
print('成功转账 %s 给卡号为 %s 的用户 %s'%(transfer_money,transfer_id,transfer_user))
temp += time.strftime("%Y %b %d %X")+'\t向卡号为 %s 的用户 %s 转账 %d \n'%(transfer_id,transfer_user,transfer_money)
else:
print('输入的卡号为 %s 的用户名为 %s 账户有误,原因可能账户信息不匹配或被冻结'%(transfer_id,transfer_user))
elif cont == '4': #查询
print('您的余额为:%.2f'%database[username]['balance'])
elif cont == '5': #打印凭条
print(temp)
elif cont == '6': #退卡
print('正在退卡。。')
time.sleep(1)
print('请取走您的银行卡,感谢使用')
ctrl_flag = True
break
else:
print('序号输入有误,可能不存在序号%s 对应的选项'%cont)
else:
count +=1
password = input('输入有误,您还有 %s 次机会\n请重新输入:'%(3-count)).strip()
else:
count +=1
password = input('密码仅为6位数字,您还有 %s 次机会\n请重新输入: '%(3-count)).strip()
with open('log.txt','a') as f1: #写入日志
f2 = open('database.txt','w')
f2.write(str(database))
f1.write(str(time.strftime("%Y %b %d %X"))+'\t修改数据库文件\n')
f2.close()
日志文件log.txt(自动生成,初始状态为空)
模拟的银行卡文件card.txt
{'yang':{'password':'111111','ID': '6662008891880688689'}}
数据文件database.txt
{'ling': {'password': '123456', 'ID': '6662008891880688688', 'balance': 500.86}, 'liu': {'password': '888888', 'ID': '6662008891880688687', 'balance': 200000.0}, 'tian': {'password': '00000', 'ID': '6662008891880688680', 'balance': 448.03}, 'liang': {'password': '666666', 'ID': '6662008891880688681', 'balance': 522.5}, 'yang': {'password': '111111', 'ID': '6662008891880688689', 'balance': 21310.0}}
结果测试:
运行环境随便你了

(截图只展示了部分,全部效果自己去体验吧)
总结优化:
1.其实不光是这个ATM机,前面的几个项目里都可以用函数或者类来实现功能,如果你们感兴趣可以用函数或者类来操作。(为什么不用函数的原因后期有空再说)。其实我现在都还正在优化中,尽量的简化代码但功能又只增不减那种
2.相信聪明的你已经发现了,程序还没有加入取款当天内只能取多少,存款存多少(单次最多存取多少)的限制,这个就自己去优化了
3.可以优化和扩展的功能还有很多,你可以做成图形化界面,也可以编译成exe可执行文件,尽量的做到和真实的ATM机一样的功能,那你就牛逼了
洗礼灵魂,修炼python(77)--全栈项目实战篇(5)—— ATM自动存取机系统的更多相关文章
- 洗礼灵魂,修炼python(82)--全栈项目实战篇(10)—— 信用卡+商城项目(模拟京东淘宝)
本次项目相当于对python基础做总结,常用语法,数组类型,函数,文本操作等等 本项目在博客园里其他开发者也做过,我是稍作修改来的,大体没变的 项目需求: 信用卡+商城: A.信用卡(类似白条/花呗) ...
- 洗礼灵魂,修炼python(73)--全栈项目实战篇(1)——【转载】前提准备之学习ubuntu
本篇是为项目实战做准备,学习Linux是必备的,不然都不好意思叫全栈对吧?下面是一位资深大神写的文章,够详细,我也不用浪费时间再写了 原文链接:Ubuntu学习——第一篇 内容: 一. Ubuntu简 ...
- 洗礼灵魂,修炼python(78)--全栈项目实战篇(6)—— 多级目录菜单之地址管理系统
相信各位都在在网上买过东西吧?那么今天我的主题就是写个在线购物系统,是不可能的,哈哈(后期确实有这个项目),那么购物都填写过快递地址吧?然后网上查个地址都有地址管理吧? 要求: 1.打印出省.市.县等 ...
- 洗礼灵魂,修炼python(86)--全栈项目实战篇(12)—— 利用socket实现文件传输/并发式聊天
由于本篇博文的项目都很简单,所以本次开个特例,本次解析两个项目,但是都很简单的 项目一:用socket实现文件传输 本项目很简单,作为小项目的预热的,前面刚学完socket,这里马上又利用socket ...
- 洗礼灵魂,修炼python(80)--全栈项目实战篇(8)—— 计算器
用正则表达式开发一个计算器,计算用户给定的一串带有加减乘除的公式. 要求:不能使用eval转换字符串 分析: 要求简单,就是计算混合运算,但是不能使用eval直接转换,主要就是把整个式子中的小括号优先 ...
- 洗礼灵魂,修炼python(79)--全栈项目实战篇(7)—— 多级目录菜单之地址管理系统升级版
要求: 1.在上一篇的地址管理系统的基础上做升级改动 2.添加增删改的功能 3.尽量的贴近生活常识中的地址管理 分析: 需求不用多说了,干就完了 相关文件源码地址:github 这次由于要有增删改的操 ...
- 洗礼灵魂,修炼python(75)--全栈项目实战篇(3)—— 账户注册登录管理系统
要求: 1.系统可以创建用户和登录用户,根据用户的输入不同,做出不同的反应(创建还是登录) 2.创建用户不能创建已存在的用户名 3.登录用户的操作最多只能有三次,超过三次冻结账户,每使用一次提示用户还 ...
- 洗礼灵魂,修炼python(81)--全栈项目实战篇(9)—— 购物商城登录验证系统
都在线购物过吧?那么你应该体验过,当没有登录账户时,点开购物车,个人中心,收藏物品等的操作时,都会直接跳转到登录账户的界面,然后如果登录一次后就不用再登录,直到用户登出. 是的,本次项目就是做一个登录 ...
- 洗礼灵魂,修炼python(76)--全栈项目实战篇(4)—— 购物车系统
要求: 1.基本符合日常购物车的要求(根据你的想法开放性提升功能) 2.展示商品信息,并且可随时上新商品 3.用户购买每一样商品时都对所剩的钱做一次对比,如果够则提示“已购买”,如果不够提示“余额不足 ...
随机推荐
- github代码clone加速
这阵子想看看开源项目 MyBatis 的源码,结果使用 git 的 clone 命令怎么也 clone 不下来,我以为是网速慢,上 Google 一搜,原来 Github 的域名被 DNS 污染了,我 ...
- 【转载】ucos临界段
其实很简单: 临界段就是不可中断的程序段,比如从UART中读取当前传递回来的值,如果有UART中断,此时这个值又会改变.同样临界段就是保护这类全局变量,如在读取时间节拍时,不应该被时钟更新时钟 ...
- Linux Namespace : User
User namespace 是 Linux 3.8 新增的一种 namespace,用于隔离安全相关的资源,包括 user IDs and group IDs,keys, 和 capabilitie ...
- “多个单核CPU”与“单个多核CPU”哪种方式性能较强?
多个单核CPU: 成本更高,因为每个CPU都需要一定的线路电路支持,这样对主板上布局布线极为不便.并且当运行多线程任务时,多线程间通信协同合作也是一个问题.依赖总线的传输,速度较慢,且每一个线程因为运 ...
- 南大算法设计与分析课程复习笔记(4)L4 - QuickSort
一.快速排序 算法导论上关于快速排序有两种写法 第一种,从头到尾遍历,不断将小于基准元素的项移到前面.代码很简介,只需要维护一个交换位置,表示小于基准元素的末尾位置加一 我们看算法导论上的一个例子: ...
- Scrapy爬虫(4)爬取豆瓣电影Top250图片
在用Python的urllib和BeautifulSoup写过了很多爬虫之后,本人决定尝试著名的Python爬虫框架--Scrapy. 本次分享将详细讲述如何利用Scrapy来下载豆瓣电影To ...
- 结构型---桥接模式(Bridge Pattern)
定义 桥接模式即将抽象部分与实现部分脱耦,使它们可以独立变化.桥接模式的目的就是使两者分离,根据面向对象的封装变化的原则,我们可以把实现部分的变化封装到另外一个类中,这样的一个思路也就是桥接模式的实现 ...
- 35.QT-多线程
程序和进程的区别 进程是动态的,程序是静态的,进程是程序运行时的实例,是占用系统运行资源的程序 进程是暂时的,程序是永久的, 进程是通过程序运行时得到的 程序是一个数据文件,进程是内存中动态的运行实体 ...
- WarShall算法
1.引言 图的连通性问题是图论研究的重要问题之一,在实际中有着广泛的应用.例如在通信网络的联通问题中,运输路线的规划问题等等都涉及图的连通性.因此传递闭包的计算需要一个高效率的算法,一个著名的算法就是 ...
- window 服务器的Tomcat 控制台日志保存到日志文件.
在Linux系统中,Tomcat 启动后默认将很多信息都写入到 catalina.out 文件中,我们可以通过tail -f catalina.out 来跟踪Tomcat 和相关应用运行的情况. ...