ATM + 购物车
需求分析
'''
- 额度 15000或自定义
- 实现购物商城,买东西加入 购物车,调用信用卡接口结账
- 可以提现,手续费5%
- 支持多账户登录
- 支持账户间转账
- 记录每月日常消费流水
- 提供还款接口
- ATM记录操作日志
- 提供管理接口,包括添加账户、用户额度,冻结账户等...
- 用户认证功能
'''
一个项目是如何从无到有的
# 1、需求分析:
'''
开发项目前,都必须找到相应的客户,让客户给企业提出项目的需求,以及需要实现的功能有哪些,拿到需求后再提取出一些列功能。
'''
# 2、设计程序以及程序的架构
'''
在所有一线开发的企业里,在一个项目开发前,都应该设计程序,那样会让程序解开耦合,从而提高项目的管理以及开发的效率。
'''
# 3、分任务开发
'''
在公司里面,开发项目需要这几种岗位人才:
UI设计: 软件的外观设计者,通过一些炫酷的设计,提高用户的对软件的体验感。
前端开发: UI设计仅仅只是把一些外观图设计出来,那前端开发需要把UI的设计图拿到之后,对软件界面的进行排版。
后端开发(python): 项目里业务以及功能的逻辑处理!
'''
# 4、项目测试
'''
测试工程师: 对后端以及前端开发好的项目进行功能和性能测试,测试的过程中出现bug就会立即让开发人员去修整,待整个项目几乎没有bug,以及性能达到项目实际的预期,就会准备上线运行。
测试分为两种:
1.黑盒测试:
通过对软件界面的功能进行测试,测试一些能让用户看到的bug。(例如穿越火线的卡箱子等等...)
2.白盒测试:
对软件进行性能测试,例如每秒钟能承受多少用户量的访问等...
'''
# 5、上线运行
'''
运维工程师(linux): 拿到前面整个项目编写完的代码,部署到服务器,上线运行!
'''
程序架构
程序目录设计
# 接下来我们写的功能都在按照这个文件目录来写,请认真阅读这个目录所对应的功能。
'''
- ATM
- conf 配置文件文件夹
- setting.py
- lib 公共方法文件夹
- common.py 公共方法文件
- interface 接口层文件夹
- user.py 用户接口文件
- bank.py 银行接口文件
- core 用户功能文件夹
- src.py 视图文件
- db 数据处理层文件夹
- db_handler.py 数据处理功能文件
- start.py 程序的入口文件,启动文件
- readme 程序的说明文件
'''
程序模块
# confs
## settings.py(主要存一些常量,主要是功能界面,商品列表,日志路径,数据路径)
import os
import sys
########
#功能展示#
#########
FUNC_MSG = {
'0': "注销",
'1': "登录",
'2': "注册",
'3': "查看余额",
'4': "转账",
'5': "还款",
'6': "取现",
'7': "查看流水",
'8': "购物",
'9': "购物车",
}
SHOP_DIST = [
['饼干', 10],
['薯片', 10],
['火腿肠', 20],
['雪糕', 10],
['别墅', 1000000]
]
# LOG_PATH = os.path
ATM_PATH = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
DB_PATH = os.path.join(ATM_PATH, 'db')
#core
## src.py(主界面,主要是和用户交互的界面)
from libs import common
from db import db_handler
from interface import user
from interface import bank
from confs import settings
#设置一个字典类型的全局变量,用于判断是否登录
user_info = {
'user': None
}
# 注销功能
def logout():
user_info['user'] = None # 将values值设置为空
print('注销成功')
# 登录模块
def login():
count = 0 #计数,连续三次注册失败,退出功能
print("欢迎来到登录功能")
while True:
username, pwd = common.input_username_pwd() # 调用输入接口,接收输入信息
flag = common.check_user(username) # 调用核对用户接口,验证用户是否已经存在
if not flag:
print('未注册')
break
flag, msg = user.login_interface(username, pwd)
if flag:
user_info['user'] = username #登录成功,赋值
print(user_info['user'])
print(msg)
break
else:
print(msg)
count += 1
if count == 3:
break
#注册模块
def register():
print("欢迎来到注册模块")
username, pwd = common.input_username_pwd()
flag = common.check_user(username) #先判断是否存在
if flag:
print("无需注册,用户已经存在")
else:
msg = user.register_interface(username, pwd) # 调用注册接口
print(msg)
@common.login_auth # 加语法糖,查看余额前先登录
def check_extra():
print("欢迎来到查看余额模块")
msg = user.check_extra(user_info.get('user')) # 调用查看余额接口
print(msg)
# 转账,需要有发起人,接收人和钱
@common.login_auth
def transfer():
print("欢迎来到转账模块")
while True:
from_username = user_info.get('user')
to_username = input('请输入你要转账的用户名:')
flag = common.check_user(to_username) #先判断用户是否存在
if flag:
money = input("请输入你要转账的金额").strip()
if not money.isdigit():
print("请输入数字")
continue
money = int(money)
msg = bank.transfer_interface(from_username, to_username, money) #调用转账接口
print(msg)
break
print("用户不存在")
@common.login_auth
#还款模块
def repay():
print("欢迎来到还款模块")
msg = bank.repay_interface(user_info['user']) #调用还款接口
print(msg)
@common.login_auth
#取现功能
def withdraw():
print("欢迎来到取现功能")
while True:
money = input("请输入取现金额:")
if not money.isdigit():
print("输入必须是数字")
continue
else:
money = int(money)
msg = bank.withdraw_interface(user_info['user'], money)#调用取现模块
print(msg)
break
@common.login_auth
#查看流水模块
def history():
print("欢迎来到查看流水模块")
msg = bank.bank_flow_interface(user_info['user'])#调用查看流水模块
print(msg)
@common.login_auth
#购物模块
def shopping():
print("欢迎来到购物模块")
while True:
for index, goods in enumerate(settings.SHOP_DIST): #用列表存储的商品输出,得到索引和有两个元素的商品小列表
print(f'{index} {goods}')
goods_n = input("请输入你要的商品编号,按q退出:")
if goods_n == 'q':
break
if not goods_n.isdigit():
print("输入有误")
continue
goods_n = int(goods_n)
goods = settings.SHOP_DIST[goods_n] # 拿到的商品是一个有两个值(第一个是商品名,第二个是价格)的列表
goods_name = goods[0] # 列表第一个元素是商品名
user_dic = db_handler.read_json(user_info['user']) # 拿到当前用户的数据字典
my_money = user_dic['extra'] # 把用户字典中的查看额度取出来
if goods[-1] <= my_money: # 如果商品金额小于额度,可以买
if goods_name in user_dic['shop_car']:
user_dic['shop_car'][goods_name] += goods[-1] # 如果我的字典里面的购物车字典有该商品,把价格加上去
else:
user_dic['shop_car'][goods_name] = goods[-1] # 如果我的字典里面的购物车字典没有该商品,把商品名加上去,价格加上去
db_handler.save_json(user_dic) # 做完修改要保存
print(f'{goods_name}加入购物车成功')
else:
print("余额不足")
break
print(f"你的购物车是{user_dic['shop_car']}") # 买完之后要打印一下
@common.login_auth
# 购物车模块
def shopping_car():
print("欢迎来到购物车模块")
while True:
user_dic = db_handler.read_json(user_info['user']) #用户数据读取
goods_dic = user_dic['shop_car'] # 拿到用户数据字典的购物车字典
cost_choice = input(f"购物车是{goods_dic},是否选择购买y/n:") # 判断是否购买
if cost_choice == 'n':
break
elif cost_choice == 'y':
cost = sum(goods_dic.values()) # 把用户购物车字典中的值求和得到总价
if cost > user_dic['extra']:
print('余额不足,支付失败')
break
user_dic['extra'] -= cost # 支付就是把用户字典的额度减去总价
db_handler.save_json(user_dic) # 操作完了保存
print("支付成功")
break
def run():
FUNC_DICT = {
'0':logout,
'1':login,
'2':register,
'3':check_extra,
'4':transfer,
'5':repay,
'6':withdraw,
'7':history,
'8':shopping,
'9':shopping_car,
}
from confs.settings import FUNC_MSG
while True:
for k, v in FUNC_MSG.items(): # 把功能列表打印出来展示给用户,包括序号和值
print(f'{k}: {v}')
func_choice = input("请输入你需要的功能,按q退出>>>>>").strip()
if func_choice == 'q':
break
if not FUNC_DICT.get(func_choice):
print('输入有误,请重新输入')
continue
func = FUNC_DICT.get(func_choice)
func()
if __name__ == '__main__':
run()
# db
## db_handler(数据处理模块,主要是用户存取数据,格式是json)
import os
import json
from confs import settings
import sys
def save_json(user_dic): #存数据,记得存的是字典,所有拼接名字的时候要把字典里面的用户名取出来
user_path = os.path.join(settings.DB_PATH,
f'{user_dic.get("username")}.json') #拼接路径
with open(user_path, 'w', encoding='utf8') as fw:
json.dump(user_dic, fw) # 存进去,第一个参数是数据,第二个是文件,就是把第一个数据存到第二个文件中
def read_json(username):# 读数据, 拼路径,读出来数据就可以了,记得返回
user_path = os.path.join(settings.DB_PATH, f'{username}.json')
if os.path.exists(user_path):
with open(user_path, 'r', encoding='utf8') as fr:
data = json.load(fr)
return data
#interface(主要就是银行和用户两个接口)
## bank.py
from db import db_handler
def transfer_interface(from_username, to_username, money):# 转账接口,
from_user_dic = db_handler.read_json(from_username) # 读当前用户数据
to_user_dict = db_handler.read_json(to_username) # 读要转账用户数据
my_money = from_user_dic['extra'] # 把用户的钱单独拿出来
if money > my_money:
return '钱不够'
else:
from_user_dic['extra'] -= money # 我减钱
to_user_dict['extra'] += money # 对方加钱
msg_f = f'已向{to_username}转账{money}元' #记录我的信息
msg_t = f'已收到{from_username}转账{money}元' #记录对方信息
from_user_dic['bank_flow'].append(msg_f) #加我流水
to_user_dict['bank_flow'].append(msg_t) #加对方流水
db_handler.save_json(from_user_dic) #存数据
db_handler.save_json(to_user_dict) #存数据
return msg_f #返回操作信息
def repay_interface(username): #还款接口
while True:
money = input("请输入你的还款金额:").strip()
if not money.isdigit():
print('请输入数字')
else:
money = int(money)
user_dic = db_handler.read_json(username) #拿用户数据
user_dic['extra'] += money # 字典里面加钱
db_handler.save_json(user_dic) #保存数据
return f'{username}已成功还款{money}' #返回信息
def withdraw_interface(username, money):# 取现接口
user_dic = db_handler.read_json(username) #拿用户信息
if money*1.005 > user_dic['extra']:
return "钱不够"
else:
user_dic['extra'] -= money*1.05 # 减钱
db_handler.save_json(user_dic) # 存数据
return f'{username}已成功取现{money}元'
def bank_flow_interface(username): # 银行流水信息
user_dic = db_handler.read_json(username) # 拿用户数据
return user_dic['bank_flow'] #返回用户字典里的流水
## user.py(用户接口)
from db import db_handler
from confs import settings
import os
def register_interface(username, pwd): #注册接口
#新生成的用户的信息字典
user_dic = {
'username': username,
'pwd': pwd,
'extra': 1500000,
'bank_flow':[],
'shop_car':{}
}
db_handler.save_json(user_dic) #保存数据
return f'{username}注册成功'
def login_interface(username, pwd): #登录模块
user_data = db_handler.read_json(username) # 拿数据
if user_data['pwd'] == pwd:
return True, '登录成功'
else:
return False, '密码输入错误'
def check_extra(username):# 查看余额接口
user_data = db_handler.read_json(username) # 拿数据
extra = user_data['extra']
return f'{username}查看了余额,余额为{extra}元'
# libs
## common.py(通用模块)
import logging
import os
from db import db_handler
from confs import settings
import logging.config
def input_username_pwd():# 用户输入的模块
username = input('请输入用户名>>>>').strip()
pwd = input("请输入密码>>>>").strip()
return username, pwd
def check_user(username): # 检查用户是否存在模块
user_path = os.path.join(settings.DB_PATH, f'{username}.json') #拼接用户路径,判断是否有该路径
if os.path.exists(user_path):
return True
else:
return False
def login_auth(func): # 装饰函数,用于验证是否登录,套用装饰函数模板
from core import src
def inner(*args, **kwargs):
if src.user_info.get('user'):
res = func(*args, **kwargs)
return res
else:
print('未登录,请去登录!')
src.login()
return inner
#start.py (开始模块,项目的起始点)
from core import src
import os
import sys
sys.path.append(os.path.dirname(__file__)) #一定要把当前路径加入环境变量里
if __name__ == '__main__':
src.run()
注意点
1. 首先要把项目单独打开,即作为根目录。
2. 记住写项目,尤其是需要导入模块时,一定要写开始的函数(start.py),可以避免出现循环导入问题,造成一些变量传入出错。
3. 有些时候Pycharm不能识别数据类型,就没有提示,但不代表是错的,只要可以运行就可以。
4. 保存的是字典,所以拼接路径时要注意名字,取字典里key = username对应的值作为名字
5. 登录成功后把user_info['user']赋值为输入的用户名