事务
 
爸妈让往他们银行卡里转点儿钱,可是我这钱全在支付宝里,爸妈又没有支付宝,只能从支付宝里转给他
们了,假如转账过程中,支付宝扣款成功了,但是银行系统崩溃了,钱没转进去,这咋整?我的大洋就这样
打水漂了?肯定不可以,为了避免这种情况的发生,就用到了事务,在转钱过程中遇到了任何差错,就回到
没转钱之前的状态,这个就叫做事务
 
事务四大特性(ACID)
以下特性纯属个人理解
原子性(Atomicity):转账前 -> 转账  -> 转账成功,以上述转账为例,转账这个动作包括【从我的支付宝扣
除money,在我爸妈的卡上增加money】,中括号里的内容要么全部执行(转账成功),如果没有全部执行就
回到转账前的状态(跟没执行一个效果),不能停留在转账的中间过程——我的支付宝扣了钱,爸妈银行卡没
多钱
 
一致性(Consistency):我的理解是能量守恒,转账前后,我的支付宝金额+我爸妈卡内金额是一致的
 
隔离性(Isolation):这个一般用在并发,两个线程的操作会互相影响的情况下,隔离性又分为若干个隔离级
别,下面具体讨论
 
永久性(Durability):只要事务提交了,就成了事实。
 
隔离级别
在说隔离级别之前先来说几个概念
 
脏读:读到了另一个事务未提交的更改
 
例如 
 事务一:包工头给农民工转账
 事务二:农民工查看工资
 
 事务一:开始事务
 事务一:包工头给农民工转账1000
 事务二:开始事务
 事务二:农民工查看账户,多了1000块
 事务二:提交
 事务一:包工头回滚,转账失败
 
不可重复读:一个事务对同一记录的读取结果不一致,因为另一个事务更新了该记录,并提交
 
例如
 事务一:查看宾馆8301 的状态
 事务二:预订8301房间
 
 事务一:开始事务
 事务一:查看8301状态,未预定
 事务二:开始事务
 事务二:预定8301房间
 事务二:提交
 事务一:再次查看8301 状态,被预定
 
幻读:一个事务执行同一个查询结果不一样,因为另一事务插入了新的记录,并提交
 
例如
 事务一:统计网站的注册用户数
 事务二:注册新的用户
 
 事务一:开始事务
 事务一:查看注册用户数为10000
 事务二:开始事务
 事务二:新增一个用户(插入一条记录)
 事务二:提交
 事务一:查看注册用户数为10001
 
四种隔离级别
 
是否允许脏读
是否不可重复读
是否幻读
read uncommitted
read committed
repeatable read
serializable
事务的使用:
MariaDB [jason]> begin;
Query OK, 0 rows affected (0.00 sec)
 
MariaDB [jason]> insert into test values(2);
Query OK, 1 row affected (0.00 sec)
 
MariaDB [jason]> select * from test;
+------+
| i    |
+------+
|    1 |
|    2 |
+------+
2 rows in set (0.00 sec)
 
MariaDB [jason]> rollback;
Query OK, 0 rows affected (0.00 sec)
 
MariaDB [jason]> select * from test;
+------+
| i    |
+------+
|    1 |
+------+
1 row in set (0.00 sec)
 
MariaDB [jason]> insert into test values(2);
Query OK, 1 row affected (0.00 sec)
 
MariaDB [jason]> commit;
Query OK, 0 rows affected (0.00 sec)
 
MariaDB [jason]> select * from test;
+------+
| i    |
+------+
|    1 |
|    2 |
+------+
2 rows in set (0.00 sec)
 
mysql默认自动开启事务,可以通过begin 手动开启事务,如果想要回到begin之前的状态,则rollback,
操作完成后记得commit,否则推出窗口会就等同rollback 了
 
mysql 中的锁 
 
行级别锁只对事务安全的表(innodb,bdb)有效
 
for update 行级锁
首先我们先建一张表,插入两条数据
 
create table test(
f1 int,
f2 char(10),
index (f1)
)engine=innodb;
 
insert into test (f1) values
(1),
(2);
 
假设我们现在有个需求,f1 必须是唯一的,不可重复,每次插入只能插入当前最大值加 1,假设有两个窗
口各有一个事务,在读取最大值并插入
 
窗口1:
MariaDB [jason]> begin;
Query OK, 0 rows affected (0.00 sec)
 
MariaDB [jason]> select max(f1) from test;
+---------+
| max(f1) |
+---------+
|       2 |
+---------+
1 row in set (0.00 sec)
 
MariaDB [jason]> insert into test (f1) values(3);
Query OK, 1 row affected (0.00 sec)
 
窗口2:
MariaDB [jason]> begin;
Query OK, 0 rows affected (0.00 sec)
 
MariaDB [jason]> select max(f1) from test;
+---------+
| max(f1) |
+---------+
|       2 |
+---------+
1 row in set (0.00 sec)
 
MariaDB [jason]> insert into test (f1) values(3);
Query OK, 1 row affected (0.00 sec)
 
窗口1:
MariaDB [jason]> commit;
Query OK, 0 rows affected (0.01 sec)
 
窗口2:
MariaDB [jason]> commit;
Query OK, 0 rows affected (0.01 sec)
 
现在查看数据:
MariaDB [jason]> select * from test;
+------+------+
| f1   | f2   |
+------+------+
|    1 | NULL |
|    2 | NULL |
|    3 | NULL |
|    3 | NULL |
+------+------+
4 rows in set (0.00 sec)
 
结果我们插入了重复的数据,这与我们的要求相背离了,mysql 提供了锁,可以解决这个问题
我们先把错误数据删掉
 
MariaDB [jason]> delete from test where f1=3;
Query OK, 2 rows affected (0.00 sec)
 
我们可以用一个for update 锁来达到这个目的,只能有一个会话可以拥有这个锁,当另一个会话也申请这个锁
时,会暂时卡住,直到前一个会话commit  释放锁之后,
 
窗口1:
MariaDB [jason]> begin;
Query OK, 0 rows affected (0.00 sec)
 
MariaDB [jason]> select max(f1) from test for update;
+---------+
| max(f1) |
+---------+
|       2 |
+---------+
1 row in set (0.00 sec)
 
窗口2:
MariaDB [jason]> begin;
Query OK, 0 rows affected (0.00 sec)
 
MariaDB [jason]> select max(f1) from test;
+---------+
| max(f1) |
+---------+
|       2 |
+---------+
1 row in set (0.00 sec)
 
MariaDB [jason]> select max(f1) from test for update;
窗口2 暂时没有查询结果,要等到 窗口1 commit 之后才行
 
窗口1:
MariaDB [jason]> insert into test (f1) values (3);
Query OK, 1 row affected (0.00 sec)
 
MariaDB [jason]> commit;
Query OK, 0 rows affected (0.01 sec)
 
i窗口2:
+---------+
| max(f1) |
+---------+
|       3 |
+---------+
1 row in set (0.00 sec)
 
MariaDB [jason]> insert into test (f1) values (4);
Query OK, 1 row affected (0.00 sec)
 
MariaDB [jason]> commit;
Query OK, 0 rows affected (0.01 sec)
窗口2 的等待可能会超时,报错:Lock wait timeout exceeded;
 
我们再来看看
MariaDB [jason]> select * from test;
+------+------+
| f1   | f2   |
+------+------+
|    1 | NULL |
|    2 | NULL |
|    3 | NULL |
|    4 | NULL |
+------+------+
4 rows in set (0.00 sec)
 
共享读锁:lock in share mode 行级锁
如果一个事务正在修改f1 的值,而另一个窗口想读取有关f1 的最新值,可以用共享读锁
 
窗口1:
MariaDB [jason]> begin;
Query OK, 0 rows affected (0.00 sec)
 
MariaDB [jason]> select max(f1) from test;
+---------+
| max(f1) |
+---------+
|       4 |
+---------+
1 row in set (0.00 sec)
 
MariaDB [jason]> insert into test (f1) values(5);
Query OK, 1 row affected (0.00 sec)
 
窗口2;
 
MariaDB [jason]> select max(f1) ,f2 from test lock in share mode;
上面语句会阻塞直到 窗口1 commit;同样会超时跑异常
 
表级锁
 
读锁
只能对表进行读操作,不能进行写操作,写操作被锁
 
窗口1:申请test 表的读锁,在窗口1 中可以读,写test
窗口2:允许读test,但是写操作会阻塞,直到窗口1 unlock
 
 
写锁
上锁的那段时间没有读或写操作
上锁的窗口可以执行插入和insert 操作,其他窗口执行的命令会遇到阻塞
 
lock table 不是事务安全的,在锁表之前会提交全部的活动事务,
在更新比较频繁的表中应该尽量避免表级锁,以避免拥堵
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

《一起学mysql》2的更多相关文章

  1. 《从零开始做一个MEAN全栈项目》(1)

    欢迎关注本人的微信公众号"前端小填填",专注前端技术的基础和项目开发的学习. 在本系列的开篇,我打算讲一下全栈项目开发的优势,以及MEAN项目各个模块的概览. 为什么选择全栈开发? ...

  2. 《从零开始做一个MEAN全栈项目》(2)

    欢迎关注本人的微信公众号"前端小填填",专注前端技术的基础和项目开发的学习.   上一节简单介绍了什么是MEAN全栈项目,这一节将简要介绍三个内容:(1)一个通用的MEAN项目的技 ...

  3. 《从零开始做一个MEAN全栈项目》(3)

    欢迎关注本人的微信公众号"前端小填填",专注前端技术的基础和项目开发的学习. 上一篇文章给大家讲了一下本项目的开发计划,这一章将会开始着手搭建一个MEAN项目.千里之行,始于足下, ...

  4. 《从零开始做一个MEAN全栈项目》(4)

    欢迎关注本人的微信公众号"前端小填填",专注前端技术的基础和项目开发的学习. 在上一篇中,我们讲了如何去构建第一个Express项目,总结起来就是使用两个核心工具,express和 ...

  5. 一个关于vue+mysql+express的全栈项目(一)

    最近学了mysql数据库,寻思着能不能构思一个小的全栈项目,思来想去,于是就有了下面的项目: 先上几张效果图吧       目前暂时前端只有这几个页面,后端开发方面,有登录,注册,完善用户信息,获取用 ...

  6. Vue、Nuxt服务端渲染,NodeJS全栈项目,面试小白的博客系统~~

    Holle,大家好,我是李白!! 一时兴起的开源项目,到这儿就告一段落了. 这是一个入门全栈之路的小项目,从设计.前端.后端.服务端,一路狂飙的学习,发量正在欣喜若狂~~ 接触过WordPress,H ...

  7. Vue、Node全栈项目~面向小白的博客系统~

    个人博客系统 前言 ❝ 代码质量问题轻点喷(去年才学的前端),有啥建议欢迎联系我,联系方式见最下方,感谢! 页面有啥bug也可以反馈给我,感谢! 这是一套包含前后端代码的个人博客系统,欢迎各位提出建议 ...

  8. SpringBoot 整合 Elastic Stack 最新版本(7.14.1)分布式日志解决方案,开源微服务全栈项目【有来商城】的日志落地实践

    一. 前言 日志对于一个程序的重要程度不用过多的言语修饰,本篇将以实战的方式讲述开源微服务全栈项目 有来商城 是如何整合当下主流日志解决方案 ELK +Filebeat . 话不多说,先看实现的效果图 ...

  9. 全栈项目|小书架|服务器端-NodeJS+Koa2实现首页图书列表接口

    通过上篇文章 全栈项目|小书架|微信小程序-首页水平轮播实现 我们实现了前端(小程序)效果图的展示,这篇文章来介绍服务器端的实现. 首页书籍信息 先来回顾一下首页书籍都有哪些信息: 从下面的图片可以看 ...

  10. 全栈项目|小书架|服务器开发-NodeJS 使用 JWT 实现登录认证

    通过这篇 全栈项目|小书架|服务器开发-JWT 详解 文章我们对JWT有了深入的了解,那么接下来介绍JWT如何在项目中使用. 安装 $ npm install jsonwebtoken 生成 Toke ...

随机推荐

  1. 21个Java Collections面试问答

    Java Collections框架是Java编程语言的核心API之一. 这是Java面试问题的重要主题之一.在这里,我列出了一些重要的Java集合面试问题和解答,以帮助您进行面试.这直接来自我14年 ...

  2. Leetcode练习题 Palindrome Number

    9. Palindrome Number Question: Determine whether an integer is a palindrome. An integer is a palindr ...

  3. Unity BehaviorDesigner行为树基础总结

    BehaviorDesigner——行为树,用于控制和实现AI逻辑,类似于这样: 上面这个行为树实现了这样的逻辑: 当Player有Input时按照Input值来移动,无Input时查找最近的可攻击目 ...

  4. 【zibbix自定义监控】zabbix服务自定义监控mysql的状态信息

    由于mysql我安装在zabbix_server服务的主机上,所以下面操作在zabbix服务主机上进行,注意服务主机已经安装了监控服务 实现步骤: 1.修改 zabbix_agentd.conf,添加 ...

  5. 13. 罗马数字转整数(C#)

    看到这道题,存在键值对,所以先建个泛型字典,把键值填进去. 由于这道题存在两个字符表示一个数字的情况,所以在for循环的时候判断一下,看看当前字符串中循环到的字符是否和下一个字符能够组成存在在字典里的 ...

  6. Java性能之优化RPC网络通信

    服务框架的核心 大型服务框架的核心:RPC通信 微服务的核心是远程通信和服务治理 远程通信提供了服务之间通信的桥梁,服务治理提供了服务的后勤保障 服务的拆分增加了通信的成本,因此远程通信很容易成为系统 ...

  7. 设计模式 - Java中单例模式的6种写法及优缺点对比

    目录 1 为什么要用单例模式 1.1 什么是单例模式 1.2 单例模式的思路和优势 2 写法① - 饥饿模式 2.1 代码示例 2.2 优缺点比较 3 写法② - 懒惰模式 3.1 代码示例 3.2 ...

  8. [b0037] python 归纳 (二二)_多进程数据共享和同步_管道Pipe

    # -*- coding: utf-8 -*- """ 多进程数据共享 管道Pipe 逻辑: 2个进程,各自发送数据到管道,对方从管道中取到数据 总结: 1.只适合两个进 ...

  9. React教程:4 个 useState Hook 示例

    摘要: React示例教程. 原文:快速了解 React Hooks 原理 译者:前端小智 到 React 16.8 目前为止,如果编写函数组件,然后遇到需要添加状态的情况,咱们就必须将组件转换为类组 ...

  10. yum工具及源码包

    目录 yum工具及源码包 yum yum源 yum实战案例 yum全局配置文件 制作本地yum仓库 构建企业级yum仓库 源码包 yum工具及源码包 yum yum是RedHat以及CentOS中的软 ...