Java自学者论坛

 找回密码
 立即注册

手机号码,快捷登录

恭喜Java自学者论坛(https://www.javazxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,会员资料板块,购买链接:点击进入购买VIP会员

JAVA高级面试进阶训练营视频教程

Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程Go语言视频零基础入门到精通Java架构师3期(课件+源码)
Java开发全终端实战租房项目视频教程SpringBoot2.X入门到高级使用教程大数据培训第六期全套视频教程深度学习(CNN RNN GAN)算法原理Java亿级流量电商系统视频教程
互联网架构师视频教程年薪50万Spark2.0从入门到精通年薪50万!人工智能学习路线教程年薪50万大数据入门到精通学习路线年薪50万机器学习入门到精通教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程MySQL入门到精通教程
查看: 490|回复: 0

SQLAlchemy 嵌套事务的解决方案

[复制链接]
  • TA的每日心情
    奋斗
    2024-11-24 15:47
  • 签到天数: 804 天

    [LV.10]以坛为家III

    2053

    主题

    2111

    帖子

    72万

    积分

    管理员

    Rank: 9Rank: 9Rank: 9

    积分
    726782
    发表于 2021-5-6 16:58:47 | 显示全部楼层 |阅读模式
     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

    哎...今天够累的,签到来了1...
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|手机版|小黑屋|Java自学者论坛 ( 声明:本站文章及资料整理自互联网,用于Java自学者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2025-1-23 00:52 , Processed in 0.058209 second(s), 30 queries .

    Powered by Discuz! X3.4

    Copyright © 2001-2021, Tencent Cloud.

    快速回复 返回顶部 返回列表