基于Python实现一个简单的银行转账操作


Posted in Python onMarch 06, 2016

前言

在进行一个应用系统的开发过程中,从上到下一般需要四个构件:客户端-业务逻辑层-数据访问层-数据库,其中数据访问层是一个底层、核心的技术。而且在实际开发中,数据库的操作也就是说数据访问层都是嵌套在其他语言中的,其是编程的核心。本文面向的是python语言,即通过python操作数据库来实现简单的银行转账操作。

工具

python提供了python DB API用来统一操作数据库,使访问数据库的接口规范化,在没有python DB API之前,接口程序十分混乱,不同的数据库需要不同的操作接口,所以这个接口提供了极大的方便。在具体操作的时候,我们需要操作数据库以及其他逻辑的python代码,数据库连接对象connection来建立连接,数据库交互对象cursor来“运送”数据,一个健壮的系统必不可少的便是数据库异常类Exceptions。整个访问数据库流程如下图:

基于Python实现一个简单的银行转账操作

接下来分别介绍下两个主要对象:

connection:数据库连接对象,建立python客户端与数据库的网络连接。
创建方法:MySQLdb.connect(),包括的主要成员方法:
cursor():使用该连接创建并返回游标
commit():提交当前事务
rollback():回滚当前事务
close()关闭连接
cursor:游标对象,用于执行查询与获取结果,cursor对象支持的主要方法如下:
execute():执行SQL语句,将结果从数据库获取到客户端
fetchone():取得结果集的下一行
fetchmany(size):获取结果集的下size行
fetchall():获取结果集中剩下的所有行
rowcount:最近一次execute返回数据的行数
close():关闭游标对象
在上面的方法中提到了一个关键名词:事务,什么是事务呢?他是访问和更新数据的一个程序执行单元,很多操作的一个集合,有四个特点:

原子性:事物中包括的诸操作要么都做,要么都不做
一致性:事务必须使数据库从一致性状态变到另一个一致性状态
隔离型:一个事务的执行不被其他事务干扰
持久性:事务一旦提交,它对数据库的改变就是持久性的
事务的上述特点正是我们完成银行转账操作的关键。

具体实现

在开发中我们怎么样使用事务呢?

关闭自动commit()
正常结束事务:conn.commit(),
异常结束事务:conn.rollback()
在银行转账系统中,需要考虑如下需求:比如A给B转账,当A账户上减少了M钱时,必须在B账户上多了M钱,不能A减了B没加,也不能B加了A还没有减,当然账户必须是有效的,M钱的金额肯定要大于A账户上的金额。所以在具体设计的时候,需要将A账户的金钱减少和B账户的金钱增加作为一个事务,要么同时成功,要么一起失败。按照这个需求,书写代码,详细代码见github,代码复制和数据库如下,有两个账户,分别拥有金钱110和10,在运行代码的时候在参数栏输入1,2,100(source_acctid, target_acctid, tranfer_money)。

基于Python实现一个简单的银行转账操作

整个代码的逻辑如下:首先连接数据库,之后执行逻辑,然后断开数据库连接,执行的逻辑包括检查转账双方的账户是否有效,转账金额是否多于转账人的账户余额,分别给转账双方的帐号金额发生变化。如果正常结束事务,提交修改数据库,否则回滚。

#coding:utf-8
import sys
import MySQLdb
class TransferMoney():
def __init__(self, conn):
self.conn = conn
def transfer(self, src, target, money):
try:
self.check_acct_available(src)
self.check_acct_available(target)
self.has_enough_money(src, money)
self.reduce_money(src, money)
self.add_money(target, money)
self.conn.commit()
except Exception as e:
print e
self.conn.rollback()
def reduce_money(self, src, money):
cursor = self.conn.cursor()
try:
sql = "update account set money = money - %s where acctid = %s" %(money, src)
cursor.execute(sql)
print "reduce_money: " + sql
#rs = cursor.fetchall()
if cursor.rowcount != 1:
raise Exception("the account reduce money fail")
finally:
cursor.close()
def add_money(self, target, money):
cursor = self.conn.cursor()
try:
sql = "update account set money = money + %s where acctid = %s" %(money, target)
cursor.execute(sql)
print "add_money: " + sql
#rs = cursor.fetchall()
if cursor.rowcount != 1:
raise Exception("the account add money fail")
finally:
cursor.close()
def check_acct_available(self, accit):
cursor = self.conn.cursor()
try:
sql = "select * from account where acctid = %s" %accit
cursor.execute(sql)
print "check_acct_available: " + sql
rs = cursor.fetchall()
if len(rs) != 1:
raise Exception("the account %s is not exist" %accit)
finally:
cursor.close()
def has_enough_money(self, src, money):
cursor = self.conn.cursor()
try:
sql = "select * from account where acctid = %s and money >= %s " %(src, money)
cursor.execute(sql)
print "has_enough_money: " + sql
rs = cursor.fetchall()
if len(rs) != 1:
raise Exception("the account does not have enough money")
finally:
cursor.close()
if __name__ == "__main__":
source_acctid = sys.argv[1]
target_acctid = sys.argv[2]
money = sys.argv[3]
conn = MySQLdb.connect(
host = "127.0.0.1", user = '******', passwd = '******', port = 3306, db = '******'
)
tr_money = TransferMoney(conn)
try:
tr_money.transfer(source_acctid, target_acctid, money)
except Exception as e:
print e
finally:
conn.close()

总结

通过对数据库的操作就可以实现一个简单的银行转账系统,所以在系统开发的时候,我们应该尽最大的可能,让整个系统不只是多个组件的拼接,应该实现1+1>2。

Python 相关文章推荐
python3模拟百度登录并实现百度贴吧签到示例分享(百度贴吧自动签到)
Feb 24 Python
python中的函数用法入门教程
Sep 02 Python
Python3 正在毁灭 Python的原因分析
Nov 28 Python
python实现的jpg格式图片修复代码
Apr 21 Python
Pandas 同元素多列去重的实例
Jul 03 Python
Python 3.x 判断 dict 是否包含某键值的实例讲解
Jul 06 Python
python 读取视频,处理后,实时计算帧数fps的方法
Jul 10 Python
python format 格式化输出方法
Jul 16 Python
在PyCharm中实现关闭一个死循环程序的方法
Nov 29 Python
Django 解决开发自定义抛出异常的问题
May 21 Python
python 利用toapi库自动生成api
Oct 19 Python
Python爬虫后获取重定向url的两种方法
Jan 19 Python
Python切片知识解析
Mar 06 #Python
Django Admin实现上传图片校验功能
Mar 06 #Python
python如何通过protobuf实现rpc
Mar 06 #Python
使用Python保存网页上的图片或者保存页面为截图
Mar 05 #Python
Python发送form-data请求及拼接form-data内容的方法
Mar 05 #Python
Python多线程爬虫简单示例
Mar 04 #Python
使用Python来开发Markdown脚本扩展的实例分享
Mar 04 #Python
You might like
PHP提取数据库内容中的图片地址并循环输出
2010/03/21 PHP
PHP实现163邮箱自动发送邮件
2016/03/29 PHP
javascript实现上传图片并预览的效果实现代码
2011/04/11 Javascript
js中top的作用深入剖析
2014/03/04 Javascript
javascript的解析执行顺序在各个浏览器中的不同
2014/03/17 Javascript
jQuery中click事件的定义和用法
2014/12/20 Javascript
AngularJS 工作原理详解
2016/08/18 Javascript
AngularJS 在同一个界面启动多个ng-app应用模块详解
2016/12/20 Javascript
5分钟打造简易高效的webpack常用配置
2017/07/04 Javascript
seajs实现强制刷新本地缓存的方法分析
2017/10/16 Javascript
jQuery中ajax获取数据赋值给页面的实例
2017/12/31 jQuery
微信小程序自定义可滑动日历界面
2018/12/28 Javascript
js module大战
2019/04/19 Javascript
nodejs log4js 使用详解
2019/05/31 NodeJs
vue-cli+webpack项目打包到服务器后,ttf字体找不到的解决操作
2020/08/28 Javascript
vue 表单输入框不支持focus及blur事件的解决方案
2020/11/17 Vue.js
JavaScript中的Proxy对象
2020/11/27 Javascript
JavaScript实现弹出窗口效果
2020/12/09 Javascript
利用 Chrome Dev Tools 进行页面性能分析的步骤说明(前端性能优化)
2021/02/24 Javascript
[00:23]DOTA2群星共贺开放测试 25日无码时代来袭
2013/09/23 DOTA
[31:29]完美世界DOTA2联赛PWL S3 INK ICE vs Magma 第一场 12.20
2020/12/23 DOTA
Python实现微信消息防撤回功能的实例代码
2019/04/29 Python
使用Python脚本从文件读取数据代码实例
2020/01/19 Python
详解CSS3中强大的filter(滤镜)属性
2017/06/29 HTML / CSS
欧洲最大的婴幼儿服装及内衣公司:Petit Bateau(小帆船)
2016/08/16 全球购物
统计岗位职责
2014/02/21 职场文书
《称象》教学反思
2014/04/25 职场文书
读书活动总结
2014/04/28 职场文书
经营理念口号
2014/06/21 职场文书
群众路线教育实践活动民主生活会个人检查对照思想汇报
2014/10/04 职场文书
党的群众路线调研报告
2014/11/03 职场文书
消费者投诉书范文
2015/07/02 职场文书
关于分班的感言
2015/08/04 职场文书
2019入党申请书格式和范文
2019/06/25 职场文书
MySQL8.0的WITH查询详情
2021/08/30 MySQL
Spring Cloud Netflix 套件中的负载均衡组件 Ribbon
2022/04/13 Java/Android