-- 设置数据库事务为手动的提交
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. 以太坊-Win环境下remix环境搭建

    一.node.js环境搭建 有多个安装方法,但是注意npm与node版本相关性较强!以下方案较为简便 1.下载网址 http://nodejs.cn/download/ 2.下载window 64位 ...

  2. informix 数据库锁表分析和解决方法

    一.前言 在联机事务处理(OLTP)的数据库应用系统中,多用户.多任务的并发性是系统最重要的技术指标之一.为了提高并发性,目前大部分RDBMS都采用加锁技术.然而由于现实环境的复杂性,使用加锁技术又不 ...

  3. pytest框架运用

    import pytest ''' 运行方式 1. pytest -s test01.py 把print信息打印出来运行用例 2. 通过main运行 前置后置方法 1. 函数级 setup teard ...

  4. 'utf-8' codec can't decode byte的解决办法

    参考链接:https://www.cnblogs.com/zyh19980816/p/11830065.html 问题:''utf-8' codec can't decode byte 0xa3 in ...

  5. 利用奇偶数来获取websocket推送时间间隔(或者比较前一个数和下一个数的变化)

    利用奇偶数来获取websocket推送时间间隔(或者比较前一个数和下一个数的变化) 在vue中的 data () {     return { countTime: 0,         newDat ...

  6. Go是一门什么样的语言?

    Go语言基本介绍 首先Go语言诞生于2007年由谷歌公司研发,2009年开源,2012年推出1.0版本,Go是一种语言层面支持并发(Go最大的特色.天生支持并发).内置runtime,支持垃圾回收(G ...

  7. QT从入门到入土(四)——多线程(QtConcurrent::run())

    引言 在前面对Qt多线程(QThread)做了详细的分析:QT从入门到入土(四)--多线程(QThread) - 唯有自己强大 - 博客园 (cnblogs.com) 但是最近在做项目时候,要将一个函 ...

  8. 【PTA|Python】浙大版《Python 程序设计》题目集:第二章

    前言 Hello!小伙伴! 非常感谢您阅读海轰的文章,倘若文中有错误的地方,欢迎您指出-   自我介绍 ଘ(੭ˊᵕˋ)੭ 昵称:海轰 标签:程序猿|C++选手|学生 简介:因C语言结识编程,随后转入计 ...

  9. ms17-010

    永恒之蓝和ms17-010简介: 永恒之蓝(EternalBLUE)"是Shadow Brokers(影子经纪人)黑客组织公布的一款黑客工具,该工具利用的漏洞也被称为MS17-010漏洞,M ...

  10. Android开发失业50天,面了10家公司,唯二的offer也主动拒了

    最近在论坛看到这样一个帖子: 坐标深圳. 4 月上旬公司解散.(现在想想好像是假解散,真裁员) 这一个半月以来,从朋友内推,到拉勾.Boss 直聘,再到猎聘.智联招聘. 从开始的精准投递,到后来的海投 ...