温馨提示×

温馨提示×

您好,登录后才能下订单哦!

密码登录×
登录注册×
其他方式登录
点击 登录注册 即表示同意《亿速云用户服务条款》

python django事务实例源码解析

发布时间:2021-09-17 15:11:37 来源:亿速云 阅读:176 作者:chen 栏目:云计算

这篇文章主要讲解了“python django事务实例源码解析”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“python django事务实例源码解析”吧!

python Django事务

# atomic() 方法   
# from django.db import transaction
###################
# atomic()
###################
def atomic(using=None, savepoint=True):    #  装饰器和上下文管理器必须.()调用方法,因为真正的处理是该方法返回的实例,不是该方法本身
    if callable(using):
        return Atomic(DEFAULT_DB_ALIAS, savepoint)(using)
    # Decorator: @atomic(...) or context manager: with atomic(...): ...
    else:
        return Atomic(using, savepoint)
##########################################
#  Atomic类  省略了非核心内容
############################################
class Atomic(ContextDecorator):
  def __init__(self, using, savepoint):
    self.using = using
    self.savepoint = savepoint
  def __enter__(self):
    connection = get_connection(self.using)
    sid = connection.savepoint()            # 进入with创建一个保存点
    # .............do
  def __exit__(self, exc_type, exc_value, traceback):
    if connection.in_atomic_block:
    # do.............
    if sid is not None:
        try:
            connection.savepoint_commit(sid)        #  提交事务
        except DatabaseError:
            try:
                connection.savepoint_rollback(sid)    #  捕获数据库异常回滚
                connection.savepoint_commit(sid)
            except Error:
                connection.needs_rollback = True
            raise
    ## 还有一段代码是exec_type收到其他程序异常时候 全局回滚,此处省略
    # do.................
###############################
# ContextDecorator
#################################
class ContextDecorator(object):
    def __call__(self, func):
        def inner(*args, **kwargs):
            with self:              #  把函数放进self的with上下文管理器,效果with相同,只是控制细粒度不同
                return func(*args, **kwargs)
        return inner

python MySQLdb

class Tran():
    def __init__(self, conn=None, close=True):
        if conn is None:                    # 创建数据库链接
            print 'init'
            self.conn = conn_tbkt()
            self.cur = self.conn.cursor()
            self.sql = []

    def __enter__(self):                            # 上下文管理器返回 sql语句列表  with Tran('tbkt_pxb') as sqls:
        print 'enter'
        return self.sql     # sql.append('select 1')

    def __exit__(self, exc_type, exc_val, exc_tb):
        print 'exit'
        try:

            print self.sql                              # 执行sql
            for s in self.sql:
                self.cur.execute(s)
            self.conn.commit()
        except:                                             # 可以捕获所有异常(django事务如果中间出现程序异常终止无法回滚)
            try:                    # 回滚本身也是sql执行,也有可能失败
                import traceback
                traceback.print_exc()
                print 'rollback'
                self.conn.rollback()
            except:
                print u'回滚失败'
        finally:
            self.cur.close()
            self.conn.close()

更细粒度的回滚:

# 在事务块中@atomic() 或者 with atomic():
sid = transaction.savepoint('tbkt_pxb')
try:
  # do ..........
except:
  transaction.savepoint_rollback(sid, 'tbkt_pxb')

注意:如果有多个数据库有路由,则需要指定和路由返回一致的useing: math3下的model需要事务,即使ziyuan_new和default是同一个库,也必须使用useing=ziyuan_new

        ziyuan_app = ['math3', 'ziyuan']
        if model._meta.app_label in ziyuan_app:
            return "ziyuan_new"

        return 'default'

调用时候必须.()方法调用

atomic块中必须注意try的使用,如果手动捕获了程序错误会导致atomic包装器捕获不到异常,也就不会回滚。要么try内代码不影响事务操作,要么就捕获异常后raise出,让atomic可以正常回滚(就是因为没有注意到这个问题,导致尝试了好几天都没成功,切记)

感谢各位的阅读,以上就是“python django事务实例源码解析”的内容了,经过本文的学习后,相信大家对python django事务实例源码解析这一问题有了更深刻的体会,具体使用情况还需要大家实践验证。这里是亿速云,小编将为大家推送更多相关知识点的文章,欢迎关注!

向AI问一下细节

免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。

AI