基于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 相关文章推荐
Python编码时应该注意的几个情况
Mar 04 Python
Python实现把回车符\r\n转换成\n
Apr 23 Python
在Debian下配置Python+Django+Nginx+uWSGI+MySQL的教程
Apr 25 Python
用Python编写生成树状结构的文件目录的脚本的教程
May 04 Python
Python爬虫实现获取动态gif格式搞笑图片的方法示例
Dec 24 Python
Python Pexpect库的简单使用方法
Jan 29 Python
在python中用print()输出多个格式化参数的方法
Jul 16 Python
python ctypes库2_指定参数类型和返回类型详解
Nov 19 Python
基于python实现学生信息管理系统
Nov 22 Python
完美解决pycharm 不显示代码提示问题
Jun 02 Python
Python Tricks 使用 pywinrm 远程控制 Windows 主机的方法
Jul 21 Python
python使用requests库爬取拉勾网招聘信息的实现
Nov 20 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一些有意思的小区别
2006/12/06 PHP
php daodb插入、更新与删除数据
2009/03/19 PHP
php echo, print, print_r, sprintf, var_dump, var_expor的使用区别
2013/06/20 PHP
ThinkPHP中url隐藏入口文件后接收alipay传值的方法
2014/12/09 PHP
php结合ACCESS的跨库查询功能
2015/06/12 PHP
Yii2搭建后台并实现rbac权限控制完整实例教程
2016/04/28 PHP
用表格输出1-1000之间的数字实现代码(附特效)
2013/04/21 Javascript
js实现数组转换成json
2015/06/26 Javascript
jQuery实现的个性化返回底部与返回顶部特效代码
2015/10/30 Javascript
基于jQuery实现Ajax验证用户名是否存在实例
2016/03/30 Javascript
AngularJS入门教程之Select(选择框)详解
2016/07/27 Javascript
javaScript基础详解
2017/01/19 Javascript
JavaScript中匿名函数的递归调用
2017/01/22 Javascript
nodejs发送http请求时遇到404长时间未响应的解决方法
2017/12/10 NodeJs
jquery.onoff实现简单的开关按钮功能(推荐)
2018/05/24 jQuery
详解npm 配置项registry修改为淘宝镜像
2018/09/07 Javascript
layui table单元格事件修改值的方法
2019/09/24 Javascript
CountUp.js数字滚动插件使用方法详解
2019/10/17 Javascript
JQuery表单元素取值赋值方法总结
2020/05/12 jQuery
[48:18]DOTA2-DPC中国联赛 正赛 RNG vs Dynasty BO3 第二场 1月29日
2021/03/11 DOTA
零基础写python爬虫之抓取百度贴吧代码分享
2014/11/06 Python
Python中字典映射类型的学习教程
2015/08/20 Python
关于反爬虫的一些简单总结
2017/12/13 Python
python如何获取当前文件夹下所有文件名详解
2019/01/25 Python
Python使用graphviz画流程图过程解析
2020/03/31 Python
安装python3.7编译器后如何正确安装opnecv的方法详解
2020/06/16 Python
Python截图并保存的具体实例
2021/01/14 Python
HTML5 video循环播放多个视频的方法步骤
2020/08/06 HTML / CSS
机关道德讲堂实施方案
2014/03/15 职场文书
党员批评与自我批评发言
2014/10/02 职场文书
收入及婚姻状况证明
2014/11/20 职场文书
2014年财政工作总结
2014/12/10 职场文书
中学生思想品德评语
2014/12/31 职场文书
感谢信模板大全
2015/01/23 职场文书
法院个人总结
2015/03/03 职场文书
用Python可视化新冠疫情数据
2022/01/18 Python