Mysql学习MySQL嵌套事务所遇到的问题

Mysql学习MySQL嵌套事务所遇到的问题,第1张

概述介绍《Mysql学习MySQL嵌套事务所遇到的问题》开发教程,希望对您有用。

《MysqL学习MysqL嵌套事务所遇到的问题》要点:
本文介绍了MysqL学习MysqL嵌套事务所遇到的问题,希望对您有用。如果有疑问,可以联系我们。

MysqL是支持嵌套事务的,但是没多少人会这么干的…. 前段时间在国外看到一些老外在争论MysqL嵌套事务的场景必要性. 逗死我了,这嵌套的鬼畜用法还有啥场景必要性.   跟以前的dba同事聊过,得知,在任何场景下都不要使用MysqL嵌套的事务.MysqL实例

那么使用MysqL嵌套事务会遇到什么问题 ?MysqL实例

MysqL> select * from ceshi; +------+ | n  | +------+ |  1 | +------+ 1 row in set (0.00 sec)  MysqL> start transaction ; query OK,0 rows affected (0.00 sec)  MysqL> insert into ceshi values(2); query OK,1 row affected (0.00 sec)  MysqL> start transaction ; query OK,0 rows affected (0.00 sec)  MysqL> insert into ceshi values(3); query OK,1 row affected (0.00 sec)  MysqL> commit; query OK,0 rows affected (0.00 sec)  MysqL> rollback; query OK,0 rows affected (0.00 sec) 


虽然我在最后rollback回滚了,但是数据显示是  1 2 3  .    原本大家以为我的事务虽然是嵌套的状态,但感觉最后rollback回滚了,其实我们希望看到的结果是 子事务执行成功,外层事务的失败会回滚的.  但事实不是这样的,最后的结果是  1 2 3 .
MysqL实例

+-----+ | n   | +-----+ |  1 | |  2 | |  3 | +-----+ 

当sql解释器遇到 start transaction 时候会触发commit… !!!   MysqL实例

begin_1  sql_1  begin_2  sql_2  sql_3 commit_1  rollback_1  . MysqL实例

begin_2 被执行的时候,sql_1 已经就被提交了,当你再去执行commit_1的时候,那么sql_2 和 sql_3 就被提交了.    这时候你再去rollback,一定用都没有….    因为先前都提交完了,你能回滚啥…MysqL实例

前面说过 在架构上一般很少很少有人会 嵌套使用事务,但有时候不小心被嵌套了. 我们拿python的项目来说,首先我们使用装饰器来实现事务的包装,接着数据处理 def a() 和  def b() 函数都被事务被包装起来,单纯的用a 和 b 都没关系,都是单事务.  如果 a 逻辑里又调用 b,那么会发生什么?   对的,事务嵌套了…    我想这是绝大数业务开发都会遇到的问题.MysqL实例

那么怎么规避这风险 ?  可以加锁呀….   设立一个全局锁,当子事务创建前会判断锁的状态….MysqL实例

如果你是flask的框架,可以使用 flask g 全局变量.  MysqL实例

如果是django框架,那么可以使用 thread local使用全局变量.MysqL实例

如果是tornado、gevent这种异步io架构,可以使用 fd 做协程变量的关联.MysqL实例

@decoratordef with_transaction(f,*args,**kwargs):   db = connection.get_db_by_table("*")  try:    db.begin()    ret = f(*args,**kwargs)    db.commit()  except:    db.rollback()    raise  return ret  @with_transactiondef hIDe(self):  '''订单不在app端显示'''  if self.status not in OrderStatus.allow_deletion_statuses():    raise OrderStatusChangeNotAllowed(self.status,OrderStatus.deleted)...  @with_transactiondef change_receipt_info(self,address,name,phone):  region = Region.get_by_address(address)  ...

当我们去执行下面语句的时候,事务会被强制提交.   当然这里前提是 autocommit = True .MysqL实例

ALTER FUNCTION  ALTER PROCEDURE  ALTER table  BEGIN  CREATE DATABASE  CREATE FUNCTION  CREATE INDEX  CREATE PROCEDURE  CREATE table  DROP DATABASE  DROP FUNCTION  DROP INDEX  DROP PROCEDURE  DROP table  UNLOCK tableS  LOAD MASTER DATA  LOCK tableS  REname table  TruncATE table  SET autoCOMMIT=1  START TRANSACTION  
总结

以上是内存溢出为你收集整理的Mysql学习MySQL嵌套事务所遇到的问题全部内容,希望文章能够帮你解决Mysql学习MySQL嵌套事务所遇到的问题所遇到的程序开发问题。

如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/sjk/1163066.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-06-01
下一篇 2022-06-01

发表评论

登录后才能评论

评论列表(0条)

保存