sqlachemy 是python的orm框架,在使用一段时间后,我们通常会出现事务嵌套的情况,看到很多人写代码的时候,居然是session到处传递,这无疑是加大了代码之间的耦合度。
案例:
def save(session):
# TODO def update(session):
# TODO def service():
session = getSession();
try:
save(session);
update(session);
session.commit();
except Exception as e:
session.rollback();
finally:
if not session:
session.close(); 假设save和update是同一个事务,但是上述的实践缺强制了save和update的 session相耦合来达成,好的实践应该是:save和update无任何关系,只是在实现业务逻辑时,组合到一个事务,确保事务性即可,即: def service():
try:
save();
update();
except Exception as e:
# TODO
因此如何解决这个问题是本文需要阐述的主题。

解决方案是:

ACID是事务的四个基本特征,通常我们的理解事务是一个操作单元,要么一起成功要么一起失败(原子性);通过一个例子来直接说明如何解决的。
场景:注册用户(添加一个用户,会未用户送10个积分)

创建表

create table user(
id int not null auto_increment,
name varchar(255) not null default '' comment '用户名',
created_at datetime not null default current_timestamp comment '创建时间',
updated_at datetime not null default current_timestamp comment '更新时间',
primary key (`id`)
)engine innodb charset=utf8 comment '用户表'; create table user_credits(
id int not null auto_increment,
user_id int not null default 0 comment '用户ID',
user_name varchar(255) not null default '' comment '用户名',
score int not null default 0 comment '积分',
created_at datetime not null default current_timestamp comment '创建时间',
updated_at datetime not null default current_timestamp comment '更新时间',
primary key (`id`),
unique key `uk_user_id` (`user_id`)
)engine innodb charset=utf8 comment '用户积分表';

sqlalchemy 实现

# -*- coding:utf-8 -*-

import datetime
from sqlalchemy import create_engine, Column, Integer, String, DateTime
from sqlalchemy.ext.declarative import declarative_base
from sqlalchemy.orm import sessionmaker, scoped_session engine = create_engine("mysql+pymysql://root:root@localhost:3306/csdn", echo=True) # 必须使用scoped_session,域session可以将session进行共享
DBSession = scoped_session(sessionmaker(bind=engine)) BaseModel = declarative_base() # ----------- Relation Model Object---------------- # class User(BaseModel): __tablename__ = "user" id = Column(Integer, primary_key=True)
name = Column(String)
created_at = Column(DateTime, default=datetime.datetime.now)
updated_at = Column(DateTime, default=datetime.datetime.now) class UserCredits(BaseModel): __tablename__ = "user_credits" id = Column(Integer, primary_key=True)
user_id = Column(Integer)
user_name = Column(String)
score = Column(Integer)
created_at = Column(DateTime, default=datetime.datetime.now)
updated_at = Column(DateTime, default=datetime.datetime.now) # ----------- Service implements---------------- # def add_user(user):
" 添加用户 "
session = DBSession()
try:
session.add(user)
session.commit()
except Exception as e:
session.rollback()
print("AddUser: ======={}=======".format(e))
finally:
if not session:
session.close() def add_user_credits(userCredits, interrupt=True):
" 添加用户积分记录 "
session = DBSession()
try:
if interrupt:
raise Exception("--- interrupt ---") session.add(userCredits)
session.commit()
except Exception as e:
session.rollback()
print("AddUserCredits: ======={}=======".format(e))
finally:
if not session:
session.close() def regist_user(): session = DBSession()
try:
# 开启子事务
session.begin(subtransactions=True) # TODO Service
user = User(name='wangzhiping')
add_user(user)
add_user_credits(UserCredits(
user_id=user.id,
user_name=user.name,
score=10
), False) session.commit()
except Exception as e:
session.rollback()
print("AddUserCredits: ======={}=======".format(e))
finally:
if not session:
session.close() # ---------- exec -----------
regist_user()
 1,设置session时,需要指定为scoped_session,目的是session可以共享(ThreadLocal);
2,session.begin(subtransactions=True) 开启子事务管理;
这是实际上regist_user是在同一个线程中的session,这是add_user,add_user_credits实际上session是同一个,所以可以实现。其实这个可以更进一步扩展,把事务隔离级别,传播属性,这里不做介绍 ---------------------
作者:紫守笨
来源:CSDN
原文:https://blog.csdn.net/program_red/article/details/55194130?utm_source=copy
版权声明:本文为博主原创文章,转载请附上博文链接!

转载:https://blog.csdn.net/program_red/article/details/55194130

SQLAlchemy 嵌套事务的解决方案的更多相关文章

  1. 一个非侵入的Go事务管理库——工作原理

    在上一篇文章"一个非侵入的Go事务管理库--如何使用"中,我讲述了如何使用事务库.有些读者可能读过"清晰架构(Clean Architecture)的Go微服务: 事物管 ...

  2. sqlalchemy中文乱码问题解决方案

    本文参考http://firefish.blog.51cto.com/298258/112794/的解决方案 问题: 本文在Ubuntu上利用scrapy抓取数据写入mysql数据库时,用到sqlal ...

  3. PHP中实现MySQL嵌套事务的两种解决方案

    PHP中实现MySQL嵌套事务的两种解决方案 一.问题起源 在MySQL的官方文档中有明确的说明不支持嵌套事务: Transactions cannot be nested. This is a co ...

  4. sqlalchemy 查询结果转json个人解决方案

    参考了网上很多资料,自己搞了一个适合的 在model 内增加一个函数: class User(db.Model): __tablename__ = 'user' userid = db.Column( ...

  5. 脱离 Spring 实现复杂嵌套事务,之一(必要的概念)

    事务传播行为种类 Spring在TransactionDefinition接口中规定了7种类型的事务传播行为, 它们规定了事务方法和事务方法发生嵌套调用时事务如何进行传播: 表1事务传播行为类型 事务 ...

  6. flask-sqlalchemy分表解决方案

    转自:http://ju.outofmemory.cn/entry/61448 关键词: flask-sqlalchemy, sqlalchemy, 分表,分库 大型系统.海量数据肯定涉及到分库分表这 ...

  7. 【事务】<查询不到同一调用方法其它事务提交的更新>解决方案

    最近遇到一个很棘手的问题,至今也解释不清楚原因,不过已经找到了解决方案. 先来看看Propagation属性的值含义,@Transactional中Propagation属性有7个选项可供选择: Pr ...

  8. svn 集成 redmine 账户验证的终极解决方案

    svn 集成 redmine 账户验证的终极解决方案 赖勇浩(http://laiyonghao.com) 动机 对于大部分开发团队来说,一般都需要一套 SCM 系统,通常是 svn + redmin ...

  9. Memcache,redis,rabbitMQ,SQLAlchemy

    Memcached Memcached 是一个高性能的分布式内存对象缓存系统,用于动态Web应用以减轻数据库负载.它通过在内存中缓存数据和对象来减少读取数据库的次数,从而提高动态.数据库驱动网站的速度 ...

随机推荐

  1. DLL加载,设置相对路径

    DLL加载,设置相对路径 1. 加载dll方法之一:(./ 代表当前目录,../ 代表上层目录)包含头文件的相对路径(当前路径为源代码路径,路径 “../../” 当前项目文件夹上级目录),链接lib ...

  2. WPF: 实现带全选复选框的列表控件

    本文将说明如何创建一个带全选复选框的列表控件.其效果如下图:     这个控件是由一个复选框(CheckBox)与一个 ListView 组合而成.它的操作逻辑: 当选中“全选”时,列表中所有的项目都 ...

  3. Integer Partition(hdu4658)2013 Multi-University Training Contest 6 整数拆分二

    Integer Partition Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) T ...

  4. C#设计模式之十外观模式(Facade Pattern)【结构型】

    一.引言 快12点半了,要开始今天的写作了.很快,转眼设计模式已经写了十个了,今天我们要讲[结构型]设计模式的第五个模式,该模式是[外观模式],英文名称是:Facade Pattern.我们先从名字上 ...

  5. js 元素大小缩放实例

    元素大小缩放是一套连贯事件,按下鼠标不放,拖动鼠标 然后松开. 按下鼠标事件 当按下鼠标时,记录元素大小.鼠标按下的位置.状态位. 拖动鼠标事件 当鼠标拖动时,计算元素调用后的大小. 元素调整后大小 ...

  6. js正则表达式之人民币匹配

    人民币格式匹配 小写格式:¥ 符号 和 整数值 与小数3部分组成. (0)代码与运行结果 { // 匹配人民币 let [reg, info, rmb, result] = [ /^(¥)(-?[0- ...

  7. 玩游戏 学Flex布局

    大家好,今天推荐一个学习Flex布局的网页小游戏,非常不错,是国外的牛人开发的,值得一试,重复几次,就会大概了解Flex的属性了! 地址: http://flexboxfroggy.com/#zh-c ...

  8. [CSS] Transitions动画效果(1)

    Transitions动画效果(1) 源码 https://github.com/YouXianMing/CSS-Animations/tree/master/Transitions 效果 细节

  9. JAVA 利用MyEclipse结合TestNG测试框架进行单元测试

    利用MyEclipse结合TestNG测试框架进行单元测试   by:授客 QQ:1033553122 测试环境 jdk1.8.0_121 myeclipse-10.0-offline-install ...

  10. listview 异步加载图片并防止错位

    1.图片错位原理: 如果我们只是简单显示list中数据,而没用convertview的复用机制和异步操作,就不会产生图片错位:重用convertview但没用异步,也不会有错位现象.但我们的项目中li ...