SQLAlchemy 嵌套事务的解决方案
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 嵌套事务的解决方案的更多相关文章
- 一个非侵入的Go事务管理库——工作原理
在上一篇文章"一个非侵入的Go事务管理库--如何使用"中,我讲述了如何使用事务库.有些读者可能读过"清晰架构(Clean Architecture)的Go微服务: 事物管 ...
- sqlalchemy中文乱码问题解决方案
本文参考http://firefish.blog.51cto.com/298258/112794/的解决方案 问题: 本文在Ubuntu上利用scrapy抓取数据写入mysql数据库时,用到sqlal ...
- PHP中实现MySQL嵌套事务的两种解决方案
PHP中实现MySQL嵌套事务的两种解决方案 一.问题起源 在MySQL的官方文档中有明确的说明不支持嵌套事务: Transactions cannot be nested. This is a co ...
- sqlalchemy 查询结果转json个人解决方案
参考了网上很多资料,自己搞了一个适合的 在model 内增加一个函数: class User(db.Model): __tablename__ = 'user' userid = db.Column( ...
- 脱离 Spring 实现复杂嵌套事务,之一(必要的概念)
事务传播行为种类 Spring在TransactionDefinition接口中规定了7种类型的事务传播行为, 它们规定了事务方法和事务方法发生嵌套调用时事务如何进行传播: 表1事务传播行为类型 事务 ...
- flask-sqlalchemy分表解决方案
转自:http://ju.outofmemory.cn/entry/61448 关键词: flask-sqlalchemy, sqlalchemy, 分表,分库 大型系统.海量数据肯定涉及到分库分表这 ...
- 【事务】<查询不到同一调用方法其它事务提交的更新>解决方案
最近遇到一个很棘手的问题,至今也解释不清楚原因,不过已经找到了解决方案. 先来看看Propagation属性的值含义,@Transactional中Propagation属性有7个选项可供选择: Pr ...
- svn 集成 redmine 账户验证的终极解决方案
svn 集成 redmine 账户验证的终极解决方案 赖勇浩(http://laiyonghao.com) 动机 对于大部分开发团队来说,一般都需要一套 SCM 系统,通常是 svn + redmin ...
- Memcache,redis,rabbitMQ,SQLAlchemy
Memcached Memcached 是一个高性能的分布式内存对象缓存系统,用于动态Web应用以减轻数据库负载.它通过在内存中缓存数据和对象来减少读取数据库的次数,从而提高动态.数据库驱动网站的速度 ...
随机推荐
- 从零开始学安全(二十二)●PHP日期date参数表
$date=new DateTime(); echo $date->format("Y-m-d h:i:s");
- MySQL添加列、删除列,创建主键等常用操作总结
一. 列常用操作 ① 添加新的一列test_column,并将其作为主键,FIRST将其放在表中第一行,auto_increement是自动增长 alter table test_table add ...
- python学习笔记(二)、字符串操作
该一系列python学习笔记都是根据<Python基础教程(第3版)>内容所记录整理的 1.字符串基本操作 所有标准序列操作(索引.切片.乘法.成员资格检查.长度.最小值和最大值)都适用于 ...
- spring_05装配bean
一.前言 <bean id="user1" scope="singleton" init-method="myInit" destro ...
- springMVC_02hello案例
1.导入jar包 commons-logging-1.1.1.jar jackson-annotations-2.5.4.jar jackson-core-2.5.4.jar jackson-data ...
- Java web.xml笔记
Javaweb项目中, web.xml文件其中的各种设置, 就是简单的标注 <?xml version="1.0" encoding="UTF-8"?&g ...
- springboot之单元测试
springboot在写完之后,肯定都需要进行单元测试,如下给出一些样例 工程层次结构如图 代码如下: controller: package com.rookie.bigdata.controlle ...
- PHP 无限极分类下拉列表实现
1. 递归实现下拉列表 /** * @param 递归 实现下拉列表 分类 */ include('db.inc.php'); function getList($pid = 0,&$resu ...
- BZOJ 1874: [BeiJing2009 WinterCamp]取石子游戏(SG函数)
Time Limit: 5 Sec Memory Limit: 162 MBSubmit: 871 Solved: 365[Submit][Status][Discuss] Description ...
- cf24D. Broken robot(高斯消元)
题意 题目链接 Sol 今天上午的A题.想出来怎么做了但是没时间写了qwq 思路很简单,首先把转移方程列一下,发现每一个位置只会从下一行/左右转移过来,而且第N行都是0,那么往下转移的都可以回带. 剩 ...