一、要求

二、思路

1.购物类buy

接收 信用卡类 的信用卡可用可用余额,

返回消费金额

2.信用卡(ATM)类

接收上次操作后,信用卡可用余额,总欠款,剩余欠款,存款

其中: 1.每种交易类型不单独处理金钱,也不单独记录流水账,每种交易类型调用处理金钱的函数(传入交易类型,交易金额)
    2.处理金钱的函数,调用配置文件中关于每种交易类型的加减钱和利率

返回本次操作后信用卡可用余额,总欠款,剩余欠款,存款

3.客户端

银行管理员注册登陆
普通用户注册登陆
发送需求:注册、登陆、交易类型、交易金额

4.服务器端

调用购物类,创建购物对象(购物接口)
调用信用卡(ATM)类,处理还款,转账等操作,对利息按月记录,写入文件

5.定时任务

定时执行程序,以计算利息。

三、代码

3.1配置文件

import os

BASE_DIR = os.path.dirname(os.path.dirname(__file__))  #配置文件的上层目录
DB_DIR=os.path.join(BASE_DIR,'db') #数据文件夹
ADMIN=os.path.join(DB_DIR,'admin')
ALL_USERS=os.path.join(DB_DIR,'allusrs')
A=os.path.join(BASE_DIR,'db','s')
LOG=os.path.join(BASE_DIR,'log') TRANSACTION={
'repay':{'action':'plus','interest':0}, #还款
'withdraw':{'action':'minus','interest':0.05},#取现
'transfer':{'action':'minus','interest':0.05},#转账
'consume':{'action':'minus','interest':0},#消费
'saving':{'action':'plus','interest':0} #存款
}

3.2公共类

3.2.1购物类

class buy:
goods=[
{"name": "电脑", "price": 1999},
{"name": "鼠标", "price": 10},
{"name": "游艇", "price": 20},
{"name": "美女", "price": 998},
] def __init__(self,money,consumption,shopping_cart,):
self.money=money
self.consumption=consumption
self.shopping_cart=shopping_cart def gouwu(self): #购物模块
print('您的当前余额为:%d' %self.money)
num=int(input('请输入商品序号:'))
num-=1
if self.goods[num]["name"] in self.shopping_cart.keys(): #goods[num]["name"]取商品名
self.shopping_cart[self.goods[num]["name"]]['n']+=1 #商品数量+1
else:
self.shopping_cart[self.goods[num]["name"]]={"price":self.goods[num]["price"],'n':1,} # 创建购物车字典 {keys{"price":价格,数量:1}}
self.money-=self.shopping_cart[self.goods[num]["name"]]["price"]*self.shopping_cart[self.goods[num]["name"]]['n'] #单价*数量
self.consumption+=self.shopping_cart[self.goods[num]["name"]]["price"]*self.shopping_cart[self.goods[num]["name"]]['n'] def yichu(self): #移除购物车模块
c=int(input(' 请输入0/1选择是否移除购物车商品, 移除请输入1:'))
if c==1:
e=int(input(' 请输入要移除的商品序号:'))
d=self.goods[e-1]
if d in self.shopping_cart.keys(): #判断要移除的商品是否在购物车内
self.shopping_cart.remove(d) #移除商品
self.money=self.money+self.goods[self.goods.index(d)]["price"] #余额增加
self.consumption=self.consumption-self.goods[self.goods.index(d)]["price"] #消费总额减少
else:
print('商品不存在')
def chongzhi(self): #充值模块
pay=int(input('请输入充值金额'))
self.money=self.money+pay
print('您的当前余额为:%d' % self.money) #显示当前余额 def main(self):
print('商品清单:')
for m,n in enumerate(self.goods,1):
print(m)
for v in n.values():
print(v)
print('=============')
#消费总额清零
self.consumption=0
buy=True #定义默认一直购物
while buy:
price=0 #定义初始价格
b=1 #定义默认不退出购物或充值状态
if self.money>=price: #消费模块;金钱大于货物价格时,才能开始购物
while self.money>=price:
#计价模块,有钱就可以一直购物
self.gouwu()
#移除购物车商品模块
self.yichu()
if self.money>=0:
print('您的当前余额为:%d' %self.money) #显示当前余额
b=int(input(' 请输入0/1选择是否继续购物, 购物请输入1:'))
if b==0: #
break #退出计价模块
if b==0: #如果不购物
break #不购物退出整个购物程序
#充值模块
else:
while self.money<price: #金钱不足,可多次充钱,直到能买得起货物
a=int(input(' 您的余额不足,请输入0/1选择是否充值,充值请输入1:'))
if a==1:
self.chongzhi()
else:
break #退出充值模块
if a==0:
break #不充值退出程序
#打印购物车商品名、商品价格、消费总额、余额
print('您的消费清单为:')
for m,n in self.shopping_cart.items():
print(m,n['price'],n['n'])
#打印消费清单
print('=============')
print('您的当前余额为:%d,您的消费总额为:%d' % (self.money,self.consumption) ) #打印消费总额
return self.consumption

3.2.2 信用卡ATM类

class Atm:
credit=15000 #信用卡额度
def __init__(self,balance,debt,remaining_debt,interest,saving,id):
self.id=id #信用卡id
self.balance=balance #信用卡可用金额
self.debt=debt #总欠款
self.remaining_debt=remaining_debt #剩余欠款
self.interest=interest #手续费
self.saving=saving #存款
self.now_time=time.strftime("%Y-%m-%d %H:%M:%S")
self.now_data=time.strftime("%Y-%m")
self.struct_time=time.gmtime(time.time())
if self.struct_time.tm_mday>22:
self.now_data=self.struct_time.tm_year+'-'+str(int(self.struct_time.tm_mon)+1) def account_info(self):#打印账户信息
return '账户id%s 信用卡额度%s;信用卡可用金额%s;剩余欠款%s;'%(self.id,self.credit,self.balance,self.remaining_debt,)
def ret_account_info(self):
return [self.id,self.credit,self.balance,self.debt,self.remaining_debt,self.interest]
def repay(self,amount):#还款
self.handel_money('repay',amount)
def withdraw(self,amount): #取现
self.handel_money('withdraw',amount)
def transfer(self,amount): #转账
self.handel_money('transfer',amount)
def consume(self,amount): #消费
self.handel_money('consume',amount)
def saves(self,amount):
self.handel_money('saving',amount)
def transaction(self,a,amount):
dic={
'1':self.repay,
'2':self.withdraw,
'3':self.transfer,
'4':self.consume,
'5':self.saves
}
print("debug: a:",type(a),"amount:",type(amount)) print(a)
print(dic[a])
print(dic["5"])
dic[a](amount)
print("end debug") def handel_money(self,transaction,amount): #交易类型,
amount=int(amount)
interest=amount*settings.TRANSACTION[transaction]['interest'] #手续费计算
if settings.TRANSACTION[transaction]['action']=='plus': if amount<=self.remaining_debt:
self.remaining_debt-=amount
self.balance+=amount
else:
self.balance+=self.remaining_debt
self.remaining_debt=0
self.saving+=amount-self.remaining_debt
else: if self.saving<amount:
self.saving=0
a=amount-self.saving
self.balance-=a+interest-self.saving
# self.debt+=amount+interest
self.remaining_debt+=a+interest
a='time:%s id:%s transaction: %s amount:%s interest %s \n'%(self.now_time,self.id,transaction,amount,interest)
print(a)
mulu=os.path.join(settings.ALL_USERS,self.id)
path_name_liushui=os.path.join(mulu,str(self.id)+'name_liushui',str(self.now_data)) with open(path_name_liushui,'a')as f: #记录流水信息
f.write(a) s=[self.balance,self.debt,self.remaining_debt,self.interest,self.saving,] #更新基本信息
path_name_base=os.path.join(mulu,str(self.id)+'name_base')
pickle.dump(s,open(path_name_base,'wb'))

 3.3服务器端:

#!/usr/bin/env python
# -*- coding: utf-8 -*- import sys,os
import hashlib
import pickle
import time
import socketserver
sys.path.append(os.path.dirname(os.path.dirname(__file__)))
from config import settings
from lib import modules
from lib.modules import * class Myserver(socketserver.BaseRequestHandler): def md5(self,pwd):
'''
对密码进行加密
:param pwd: 密码
:return:
'''
hash=hashlib.md5(bytes('xx7',encoding='utf-8'))
hash.update(bytes(pwd,encoding='utf-8'))
return hash.hexdigest() def login(self,usrname,pwd,x):
'''
登陆
:param usrname: 用户名
:param pwd: 密码
:return:是否登陆成功
'''
conn=self.request
if x=='1':
path_name_pwd=os.path.join(settings.ADMIN,usrname)
else:
mulu=os.path.join(settings.ALL_USERS,usrname)
path_name_pwd=os.path.join(mulu,usrname+'name_pwd')
s=pickle.load(open(path_name_pwd,'rb'))
if usrname in s:
if s[usrname]==self.md5(pwd): #和加密后的密码进行比较
return True
else:
return False
else:
return False def regist(self,usrname,pwd,x):
'''
注册
:param usrname: 用户名
:param pwd: 密码
:return:是否注册成功
''' conn=self.request
if x=='1':
mulu=os.path.join(settings.ADMIN,usrname)
else:
mulu=os.path.join(settings.ALL_USERS,usrname) if os.path.exists(mulu):
return False
else:
os.mkdir(mulu)
s={}
s[usrname]=self.md5(pwd)
path_name_pwd=os.path.join(mulu,usrname+'name_pwd')
pickle.dump(s,open(path_name_pwd,'wb'))
path_name_base=os.path.join(mulu,usrname+'name_base')
pickle.dump([15000,{},0,0,0],open(path_name_base,'wb'))
path_name_liushui=os.path.join(mulu,usrname+'name_liushui')
os.mkdir(path_name_liushui)
return True def user_identity_authentication(self,usrname,pwd,ret,x):
'''
判断注册和登陆,并展示用户的详细目录信息,支持cd和ls命令
:return:
'''
conn=self.request
if ret=='1':
r=self.login(usrname,pwd,x)
if r:
conn.sendall(bytes('y',encoding='utf-8'))
else:
conn.sendall(bytes('n',encoding='utf-8'))
elif ret=='2':
# print(usrname,pwd)
if x=='1':
r=self.regist(usrname,pwd,x)
else: #用户注册
s=[0,1]
pickle.dump(s,open(settings.A,'wb'))
while True:
ret=pickle.load(open(settings.A,'rb'))
if ret[0]==0:
time.sleep(30)
continue
elif ret[0]==1 or ret[0]==2:
break #默认值已更改,银行管理员已操作 if ret[0]==1: #如果管理员同意
r=self.regist(usrname,pwd,x)
else:
r=0
s=[0,0]
pickle.dump(s,open(settings.A,'wb'))
if r:
conn.sendall(bytes('y',encoding='utf-8'))
else:
conn.sendall(bytes('n',encoding='utf-8'))
def interactive(self,usrname): #进行交互
conn=self.request
while True:
c=conn.recv(1024) #接收用户交互选项
r=str(c,encoding='utf-8')
mulu=os.path.join(settings.ALL_USERS,usrname)
path_name_base=os.path.join(mulu,usrname+'name_base')
s=pickle.load(open(path_name_base,'rb')) #打印账户信息
obj=modules.Atm(s[0],s[1],s[2],s[3],s[4],usrname) #Atm对象
a=obj.account_info() #接收账户信息
conn.sendall(bytes(a,encoding='utf-8')) b=obj.ret_account_info() if r== '4':
buy_obj=modules.buy(b[2],0,{})
amount=buy_obj.main()
elif r=='q':
break
else:
s=conn.recv(1024)
amount=str(s,encoding='utf-8') obj.transaction(r,amount) pass def handle(self):
conn=self.request
x=conn.recv(1024)
x=str(x,encoding='utf-8')
conn.sendall(bytes('收到用户类别',encoding='utf-8'))
while True:
if x=='1' or x=='2':
b=conn.recv(1024)
ret=str(b,encoding='utf-8')
conn.sendall(bytes('b ok',encoding='utf-8'))
c=conn.recv(1024)
r=str(c,encoding='utf-8')
usrname,pwd=r.split(',')
print(usrname,pwd)
self.user_identity_authentication(usrname,pwd,ret,x) #登陆或注册验证
if x=='2':#普通用户身份验证成功后
self.interactive(usrname) pass break
elif x=='q':
break if __name__=='__main__':
sever=socketserver.ThreadingTCPServer(('127.0.0.1',9999),Myserver)
sever.serve_forever()

3.4 用户端

#!/usr/bin/env python
# -*- coding: utf-8 -*-
'''
本程序作为用户或银行管理员的入口,其中c=1代表银行管理员,c=2代表普通用户
'''
import pickle
import sys
import time
import os
import socket
sys.path.append(os.path.dirname(os.path.dirname(__file__)))
from config import settings
from lib import *
from lib.modules import * def login(usrname,pwd):
'''
登陆
:param usrname:用户名
:param pwd:密码
:return:是否登陆成功
'''
obj.sendall(bytes(usrname+','+pwd,encoding='utf-8'))
ret=obj.recv(1024)
r=str(ret,encoding='utf-8')
if r=='y':
return 1
else:
return 0 def regist(usrname,pwd,x):
'''
注册
:param usrname:用户名
:param pwd:密码
:return:是否注册成功
'''
obj.sendall(bytes(usrname+','+pwd,encoding='utf-8')) ret=obj.recv(1024)
r=str(ret,encoding='utf-8')
if r=='y':
return 1
else:
return 0
def user_identity_authentication(usrname,pwd,x):
'''
选择登陆或注册,展示用户的详细目录信息,支持cd和ls命令
:return:
'''
a=input('请选择1.登陆 2.注册')
obj.sendall(bytes(a,encoding='utf-8'))
obj.recv(1024)
if a=='1':
ret=login(usrname,pwd)
if ret:
print('登陆成功')
return 1
else:
print('用户名或密码错误')
return 0
elif a=='2':
ret=regist(usrname,pwd,x)
if ret:
print('注册成功')
return 1
else:
print('用户名已存在或银行管理员拒绝')
return 0
def main(x):
usrname=input('请输入用户名')
pwd=input('请输入密码')
if user_identity_authentication(usrname,pwd,x): #如果验证身份成功
if x=='1': #处理用户注册信息 while True:
s=pickle.load(open(settings.A,'rb'))
if s[1]==0:
time.sleep(30)
continue
elif s[1]==1:
while True:
a=input('用户请求注册,输入1同意,2拒绝')
if a=='1':
s=[1,0]
pickle.dump(s,open(settings.A,'wb'))
break
elif a=='2':
s=[2,0]
pickle.dump(s,open(settings.A,'wb'))
break
else:
print('输入有误')
break
else: #普通用户登陆后
interactive() #进行交互 def interactive():
while True:
a=input('请选择 1.还款 2.取现 3.转账 4.消费 5.存钱 q退出')
obj.sendall(bytes(a,encoding='utf-8'))
r=obj.recv(1024) #接收账户信息
ret=str(r,encoding='utf-8')
print(ret)
if a !='4'and a !='q':
b=input('请输入金额')
obj.sendall(bytes(b,encoding='utf-8'))
elif a=='q':
break obj=socket.socket() #创建客户端socket对象
obj.connect(('127.0.0.1',9999))
while True:
x=input('请选择1.银行管理员 2.用户 q、退出')
obj.sendall(bytes(x,encoding='utf-8'))
obj.recv(1024) #确认收到用户类别
if x=='1' or x=='2':
main(x)
break
elif x=='q':
break
else:
print('输入有误请重新输入') obj.close()

 3.5定时任务

#!/usr/bin/env python
# -*- coding:utf-8 -*-
import os,sys
import json,pickle
import time
sys.path.append(os.path.dirname(os.path.dirname(__file__)))
from config import settings def main():
card_list = os.listdir(settings.ALL_USERS)
for card in card_list:
basic_info = pickle.load(open(os.path.join(settings.ALL_USERS, card, card+'name_base')))
struct_time = time.localtime() # 循环账单列表,为每月的欠款计息。并写入到当月账单中
for item in basic_info['debt']:
interest = item['total_debt'] * 0.0005
if basic_info[4] >= interest:
basic_info[4] -= interest
else:
temp = interest - basic_info[4]
basic_info[4]=0
basic_info[0] -= temp
pickle.dump(
basic_info,
open(os.path.join(settings.ALL_USERS, card, card+'name_base'),'w')
) # 如果当前等于10号(9号之前)
# 当前余额为负值,则将值添加到账单列表中,开始计息,同时,本月可用额度恢复。
date = time.strftime("%Y-%m-%d")
if struct_time.tm_mday == 11 and basic_info[2]>0:
dic = {'date': date,
"total_debt": basic_info[2],
"balance_debt": basic_info[2],
}
basic_info[1].append(dic)
# 恢复可用额度
basic_info[0] = 15000
pickle.dump(
basic_info,
open(os.path.join(settings.ALL_USERS, card, card+'name_base'),'w')
) def run():
main()

  

  

基于python网络编程实现支持购物、转账、存取钱、定时计算利息的信用卡系统的更多相关文章

  1. Python网络编程03 /缓存区、基于TCP的socket循环通信、执行远程命令、socketserver通信

    Python网络编程03 /缓存区.基于TCP的socket循环通信.执行远程命令.socketserver通信 目录 Python网络编程03 /缓存区.基于TCP的socket循环通信.执行远程命 ...

  2. Python网络编程02 /基于TCP、UDP协议的socket简单的通信、字符串转bytes类型

    Python网络编程02 /基于TCP.UDP协议的socket简单的通信.字符串转bytes类型 目录 Python网络编程02 /基于TCP.UDP协议的socket简单的通信.字符串转bytes ...

  3. python 网络编程 TCP/IP socket UDP

    TCP/IP简介 虽然大家现在对互联网很熟悉,但是计算机网络的出现比互联网要早很多. 计算机为了联网,就必须规定通信协议,早期的计算机网络,都是由各厂商自己规定一套协议,IBM.Apple和Micro ...

  4. python网络编程——IO多路复用之select

    1 IO多路复用的概念 原生socket客户端在与服务端建立连接时,即服务端调用accept方法时是阻塞的,同时服务端和客户端在收发数据(调用recv.send.sendall)时也是阻塞的.原生so ...

  5. 图解Python网络编程

    返回目录 本篇索引 (1)基本原理 (2)socket模块 (3)select模块 (4)asyncore模块 (5)asynchat模块 (6)socketserver模块 (1)基本原理 本篇指的 ...

  6. Python网络编程之网络基础

    Python网络编程之网络基础 目录 Python网络编程之网络基础 1. 计算机网络发展 1.1. OSI七层模型 1.2. 七层模型传输数据过程 2. TCP/IP协议栈 2.1 TCP/IP和O ...

  7. Python学习(22)python网络编程

    Python 网络编程 Python 提供了两个级别访问的网络服务.: 低级别的网络服务支持基本的 Socket,它提供了标准的 BSD Sockets API,可以访问底层操作系统Socket接口的 ...

  8. Day07 - Python 网络编程 Socket

    1. Python 网络编程 Python 提供了两个级别访问网络服务: 低级别的网络服务支持基本的 Socket,它提供了标准的 BSD Sockets API,可以访问底层操作系统Socket接口 ...

  9. python网络编程基础(线程与进程、并行与并发、同步与异步、阻塞与非阻塞、CPU密集型与IO密集型)

    python网络编程基础(线程与进程.并行与并发.同步与异步.阻塞与非阻塞.CPU密集型与IO密集型) 目录 线程与进程 并行与并发 同步与异步 阻塞与非阻塞 CPU密集型与IO密集型 线程与进程 进 ...

随机推荐

  1. SVN迁移到Git的过程(+ 一些技巧)

    SVN迁移到Git的过程(+ 一些技巧) 李顺利 Key Words SVN,Git,Clone,Conversion,Tips,VCS,Pro Git 关于在VCS中SVN和Git之间的迁移(Clo ...

  2. UIView的一些基本方法 init、loadView、viewDidLoad、viewDidUnload、dealloc

    init方法 在init方法中实例化必要的对象(遵从LazyLoad思想) ‍init方法中初始化ViewController本身 loadView方法 当view需要被展示而它却是nil时,view ...

  3. 【leetcode】Rotate Image

    Rotate Image You are given an n x n 2D matrix representing an image. Rotate the image by 90 degrees ...

  4. 【leetcode】Populating Next Right Pointers in Each Node II

    Populating Next Right Pointers in Each Node II Follow up for problem "Populating Next Right Poi ...

  5. 【leetcode】First Missing Positive

    First Missing Positive Given an unsorted integer array, find the first missing positive integer. For ...

  6. Python 包管理工具解惑

    Python 包管理工具解惑 本文链接:http://zengrong.net/post/2169.htm python packaging 一.困惑 作为一个 Python 初学者,我在包管理上感到 ...

  7. Java计时器Timer和TimerTask用法

    package com.sy.game.test; import java.util.Timer; import java.util.TimerTask; public class TimeTask ...

  8. MySQL binlog 组提交与 XA(分布式事务、两阶段提交)【转】

    概念: XA(分布式事务)规范主要定义了(全局)事务管理器(TM: Transaction Manager)和(局部)资源管理器(RM: Resource Manager)之间的接口.XA为了实现分布 ...

  9. ABAP 单位转换函数

    CALL FUNCTION 'UNIT_CONVERSION_SIMPLE'         EXPORTING           input                = wa_all-btg ...

  10. Mathematics:Dead Fraction(POJ 1930)

    消失了的分式 题目大意:某个人在赶论文,需要把里面有些写成小数的数字化为分式,这些小数是无限循环小数(有理数),要你找对应的分母最小的那个分式(也就是从哪里开始循环并不知道). 一开始我也是蒙了,这尼 ...