-- 设置数据库事务为手动的提交
SET @@AUTOCOMMIT = 0;

-- 查看是否被修改
SELECT @@autocommit;

-- 查看当前的编码格式
SELECT @@character_set_results;

-- 账户表
CREATE TABLE account(
id INT PRIMARY KEY AUTO_INCREMENT,
NAME VARCHAR(20),
balance INT
);

INSERT INTO account(NAME,balance) VALUES('mzy',5000);
INSERT INTO account(NAME,balance) VALUES('jacky',5000);

SELECT * FROM account;
UPDATE account SET balance=5000;
DELETE FROM account;

-- 设置为手动提交之后,必须手动commit
COMMIT;

-- 关于事务的四种特性中的隔离性的理解:
-- 1.首先调整数据库事务隔离性的级别
-- 隔离级别 脏读 不可重复读 幻读
-- Serializable         否          否           否
-- Repeatable Read      否          否           是
-- Read Committed       否          是           是
-- Read Uncommitted     是          是           是

-- mysql默认的话: Repeatable Read
-- oracle默认的话:Read Committed

-- 修改事务的等级:最低的级别Read Uncommitted:
SET GLOBAL TRANSACTION ISOLATION LEVEL READ UNCOMMITTED;
-- 1. 测试脏读
-- 查看当前的级别:
SELECT @@global.tx_isolation;

脏读的理解:
一个事务读到了另一个事务未提交更新的数据!
假如这个事务未commit,只是当前操作可以看见,
但是未提交事务!

例如具体业务中:
买家点击货物(假定价格100元),进入付款的界面了,
但是没有执行最后的提交付款(COMMIT),但是也没有
取消订单(ROLLBACK)【注意是相当于保持这个操作】。
但是通知卖家查看余额的时候,就会发现余额多了100元,
卖家一发货,就相当于损失了100元。

SET @@AUTOCOMMIT = 0; -- 首先设置事务提交为手动

-- 我先给卖家jacky打2000,让他发货
UPDATE account SET balance=balance-2000 WHERE NAME='mzy';
UPDATE account SET balance=balance+2000 WHERE NAME='jacky';

-- jacky查余额增加了2000
SELECT * FROM account WHERE NAME='jacky';

-- 然后我取消订单进行混滚了
ROLLBACK;

-- 修改事务的等级:倒数第二级别Read Committed:
-- read committed 比 read uncommitted 多了防止脏读
SET GLOBAL TRANSACTION ISOLATION LEVEL READ COMMITTED;
-- 2. 测试不可重复读
-- 查看当前的级别:
SELECT @@global.tx_isolation;

不可重复读:
一个事务得到了另一个事务已提交更新(update)的数据!

需要和上面的脏读区分出来的是,我们的脏读是站在用户的角度来看的;
但是不可重复读,是站在银行的角度来讲的,在银行的需求中,当前的
报备和日志需要统一!!
什么意思呢?
比如在一次银行对用户的存款进行查账的操作中,操作的数据
保持一致!
例如:我们当前的正在查A用户,查到A用户的存款有3000元,
但是在我们开始记录的那一个时刻点,A用户又进行存款的操作;
A用户又存了2000元;如果没有预防不可重复读的话,那么在一次
事务的操作中,如果需要多次记录结果:第一次记录结果就为3000元;
第二次记录结果就为5000元,造成两个记录的结果不同;

意思就是,两个并发进行的事务不能相互影响,这种修改的结果,只能在
当前事务结束了之后,存入,修改!

脏读是必须避免的,但是不可重复读,有些时候也是需要的;

-- 不可重复读的测试应该是两个窗口才能进行:不可重复读原理是并发并行的事务;

-- 模拟:窗口一
-- jacky存入2000元
UPDATE account SET balance = banlance + 2000 WHERE account.`NAME` = 'jacky';

-- 模拟:窗口二(在窗口一的update操作前打开)
-- 在没有防止不可重复读的时候,这里的结果应该是修改了之后的结果
-- 在防止了不可重复读之后,这里的结果应该是修改前的,
-- 当退出当前事务之后,再次进入事务,才是修改后的结果。
SELECT account.`balance` FROM account WHERE account.`NAME` = 'jacky';







-- 修改事务的等级:倒数第三级别REPEATABLE READ:
SET GLOBAL TRANSACTION ISOLATION LEVEL REPEATABLE READ;
-- 3.测试幻读
-- 查看当前的级别:
SELECT @@global.tx_isolation;

幻读:
一个事务读到另一个事务,新插入(insert)的数据。
mysql中看不到幻读,但是能防止幻读?
只能看最高级别防止了幻读后倒推!

SET GLOBAL TRANSACTION ISOLATION LEVEL SERIALIZABLE;

-- 假如,现在银行中新加了一个用户,也是并发的事务;
-- 其中如果是显示添加前的mzy和jacky两个账户就是防止了幻读
-- 如果计算上了新加的用户,就是幻读

-- 窗口一:
SELECT COUNT(account.`NAME`) FROM account;

-- 窗口二:
-- mysql中看不到幻读
INSERT INTO account VALUES(....); -- 直接卡住,不会让你插入
-- 因为如果你插入的话,就会影响到上面的查询

-- 倒推:如果没有防止幻读的话,级别不是SERIALIZABLE,而是REPEATABLE READ
-- 就会在一次查询中并发的两个事务,产生幻读首先查询count(*)为2
-- 在insert之后,查询就为3

-- 和防止可重复读不同的是,可重复读运行用户修改(update),但是另一个事务当前不会
-- 发生变化,在重新开启的时候才会发生变化。

-- 防止幻读就很简单粗暴,当在查询的时候,就禁止用户开户(insert)

-- 隔离等级:但是安全性往往会带来性能上的缺失
-- 我觉得在一般的业务中防止脏读就可以了,可重复读和幻读都是可以接受的。

我对数据库事务的理解(MYSQL中)的更多相关文章

  1. 数据库事务系列-MySQL跨行事务模型

    说来和MySQL倒是有缘,毕业的第一份工作就被分配到了RDS团队,主要负责把MySQL弄到云上做成数据库服务.虽说整天和MySQL打交道,但说实话那段时间并没有很深入的理解MySQL内核,做的事情基本 ...

  2. 如何理解 MySQL 中的 <=> 操作符?

    问题 : 我在看以前的一个开发者的代码时看到 WHERE p.name <=> NULL 在这个查询语句中 <=>符号是什么意思啊?是不是和 =号是一样啊?还是一个语法错误啊? ...

  3. 正确理解MySQL中的where和having的区别

    原文:https://blog.csdn.net/yexudengzhidao/article/details/54924471 以前在学校里学习过SQLserver数据库,发现学习的都是皮毛,今天以 ...

  4. Sybase数据库实现等效的mysql中group_concat功能

    在MySQL中,如果想实现将分组之后的多个数据合并到一列,可以使用group_concat函数,如下图所示: 但是,在Sybase中没有这样的函数(别问我为什么使用Sybase,因为公司用的Sybas ...

  5. 如何理解MySQL中auto_increment?

    1.auto_increment用于主键自动增长.比如从1开始增长,当把第一条数据删除,再插入第二条数据时,主键值为2,不是1.

  6. 深入理解MySQL中的Redo、Undo、MVCC

    http://edu.csdn.net/course/detail/3495 http://edu.csdn.net/courses/o317_a3/云计算大数据

  7. 数据库事务中的隔离级别和锁+spring Transactional注解

    数据库事务中的隔离级别和锁 数据库事务在后端开发中占非常重要的地位,如何确保数据读取的正确性.安全性也是我们需要研究的问题.ACID首先总结一下数据库事务正确执行的四个要素(ACID): 原子性(At ...

  8. 深入理解 MySQL ——锁、事务与并发控制

    本文首发于vivo互联网技术微信公众号 mp.weixin.qq.com/s/JFSDqI5ya… 作者:张硕 本文对 MySQL 数据库中有关锁.事务及并发控制的知识及其原理做了系统化的介绍和总结, ...

  9. 举例说明MySQL中的事务

    一.场景导入 现在有一张仓库表,仓库表中记录了每一个物品的数量,还有一张用户表,用户购买产品,仓库表的产品数量减少,而用户拥有产品的数量增加. 但是如果仓库中的产品数量不足时怎么处理? 例子: #仓库 ...

随机推荐

  1. Spring RestTemplate 之exchange方法

    ●exchange方法提供统一的方法模板进行四种请求:POST,PUT,DELETE,GET (1)POST请求 String reqJsonStr = "{\"code\&quo ...

  2. vuejs第一集之:vuejs了解

    1,了解到前后端分离2,连接到vuejs3,搜集书籍: Vuejs前端开发基础与项目实战 (https://detail.tmall.com/item.htm?spm=a230r.1.14.107.6 ...

  3. js实现0ms延时定时器的几种方式

    这两天看到一篇介绍<如何实现准时的 setTimeout?>的文章,文章起源于一道面试题:有什么办法让setTimeout准时呀?具体文章内容可查看附录[1],看完之后,引起了我对setT ...

  4. Laravel使用artisan快速实现表单的登陆注册

    1. 开发环境 macOS Mojave 10.14.6 XAMPP 5.6.38 Laravel 5.2 2. 在终端,先进入到项目根目录并执行执行命令 php artisan make:auth ...

  5. 记一次错误:mid=front+(rear-front)>>1;

    设rear=6,front=4,mid=front+(rear-front)>>1; mid应该等于5的,但结果却是3. 错误原因:"+"运算符的优先级高于" ...

  6. kali操作系统添加中文输入法

    今天一通操作真心累啊.想安装搜狗输入法,百度搜索了好多 三步走:https://blog.csdn.net/qq_44110340/article/details/101382732 一顿操作猛如虎, ...

  7. 迈达斯midas Gen 2019 2.1 中文汉化安装教程

    midas Gen 2019 v2.1 for win是一款关于结构设计有限元分享的工具,分为建筑领域.桥梁领域.岩土领域.仿真领域四个大类.具有人性化的操作界面,且采用了优秀的的计算机显示技术,是建 ...

  8. 构建前端第12篇之---在Vue中对组件,变量,函数的全局引入

    张燕涛写于2020-01-16 星期two 本篇还是源于import和export的使用,昨天看es6入门 和MDN文档,大体上用法了解了,但今天看ElementUI源码的时候,看到 //src/in ...

  9. 【Azure 应用服务】Azure Function HTTP 触发后, 230秒就超时。而其他方式触发的Function, 执行5分钟后也超时,如何调整超时时间?

    问题描述 Azure Function HTTP 触发后, 230秒就超时,而其他方式触发的Function, 执行5分钟后也超时,如何调整超时时间? 问题分析 查阅官方文档,对函数应用超时持续时间有 ...

  10. SQL语句(四)联表查询

    目录 一.关联查询的分类 按年代分 按功能分 二.sql92语法的连接 语法 1. 简单应用 2. 为表起别名 3. 加入筛选 4. 加入分组 5. 三表连接 6. 非等值连接 7. 自连接 三.sq ...