需求: 有一个账户,两个人在同一时间要对此账户操作,A要对账户充值100块,B要从账户中取出100块.操作前都要先看一下账户的 余额然后再操作.

--窗口1 用户进行充值

--充值前 先查看余额

set @m=0;SELECT money into @m from account where id = 1;select @m;--看到余额后 充值100 块

update account set money = @m + 100 where id = 1;SELECT * fromaccount;----------------------------------------------------------------窗口2 用户进行取款

--取款前 先查看余额

set @m=0;SELECT money into @m from account where id = 1;select @m;--看到余额后 取款100 块

update account set money = @m - 100 where id = 1;SELECT * from account;

示例

1.锁的基本概念

当并发事务同时访问一个资源时,有可能导致数据不一致,因此需要一种机制来将数据访问顺序化,以保证数据库数据的一致性。

2.锁的基本类型

多个事务同时读取一个对象的时候,是不会有冲突的。同时读和写,或者同时写才会产生冲突。因此为了提高数据库的并发性能,通常会定义两种锁:共享锁和排它锁。

2.1共享锁(Shared Lock,也叫S锁)

共享锁(S)表示对数据进行读操作。因此多个事务可以同时为一个对象加共享锁。(如果试衣间的门还没被锁上,顾客都能够同时进去参观)

2.2排他锁(Exclusive Lock,也叫X锁)

排他锁(X)表示对数据进行写操作。如果一个事务对 对象加了排他锁,其他事务就不能再给它加任何锁了。(某个顾客把试衣间从里面反锁了,其他顾客想要使用这个试衣间,就只有等待锁从里面给打开了).

3. 实际开发中常见的两种锁:

3.1悲观锁 顾名思义,就是很悲观,每次去拿数据的时候都认为别人会修改,所以每次在拿数据的时候都会上锁,这样别人想拿这个数据就会block(阻塞)直到它拿到锁。传统的关系型数据库里边就用到了很多这种锁机制.

注意:要使用悲观锁,我们必须关闭mysql数据库的自动提交属性.因为MySQL默认使用autocommit模式,也就是说,当你执行一个更新操作后,MySQL会立刻将结果进行提交。关闭自动提交命令为:set autocommit=0;

设置完autocommit后,我们就可以执行我们的正常业务了。具体如下:

-- 0.开始事务

start transaction;

-- 1.查询账户余额

set @m = 0; -- 账户余额

select money into @m from account where id = 1 for update;

select @m;

-- 2.修改账户余额

update account set money = @m -100 where id = 1;

select * FROM account where id = 1;

-- 3. 提交事务

commit;

在另外的查询页面执行:

-- 0.开始事务

start transaction;

-- 1.查询账户余额

set @m = 0; -- 账户余额

select money into @m from account where id = 1 for update;

select @m;

-- 2.修改账户余额

update account set money = @m +100 where id = 1;

select * FROM account where id = 1;

-- 3. 提交事务

commit;

会发现当前查询会进入到等待状态,不会显示出数据,当上面的sql执行完毕提交事物后,当前sql才会显示结果.

注意1:在使用悲观锁时,如果表中没有指定主键,则会进行锁表操作.

注意2: 悲观锁的确保了数据的安全性,在数据被操作的时候锁定数据不被访问,但是这样会带来很大的性能问题。因此悲观锁在实际开发中使用是相对比较少的。

3.2 乐观锁, 顾名思义,就是很乐观,每次去拿数据的时候都认为别人不会修改,所以不会上锁,但是在更新的时候会判断一下在此期间别人有没有去更新这个数据,可以使用版本号等机制。

使用乐观锁的两种方式:

1.使用数据版本(Version)记录机制实现,这是乐观锁最常用的一种实现 方式。何谓数据版本?即为数据增加一个版本标识,一般是通过为数据库表增加一个数字类型的 “version” 字段来实现。当读取数据时,将version字段的值一同读出,数据每更新一次,对此version值加一。当我们提交更新的时候,判断数据库表对应记录 的当前版本信息与第一次取出来的version值进行比对,如果数据库表当前版本号与第一次取出来的version值相等,则予以更新,否则认为是过期数 据。

代码示例:

-- 1.查询账户余额

set @m = 0; -- 账户余额

select money into @m from account where id = 1 ;

select @m;

-- 2.查询版本号

set @version = 0; -- 版本号

select version into @version from account where id = 1 ;

select @version;

-- 3.修改账户余额

update account set money = @m -100,version=version+1 where id = 1 and version = @version;

select * FROM account where id = 1;

2.乐观锁定的第二种实现方式和第一种差不多,同样是在需要乐观锁控制的table中增加一个字段,名称无所谓,字段类型使用时间戳 (datatime), 和上面的version类似,也是在更新提交的时候检查当前数据库中数据的时间戳和自己更新前取到的时间戳进行对比,如果一致则OK,否则就是版本冲突。

悲观锁与乐观锁的优缺点:

两种锁各有其有点缺点,不能单纯的讲哪个更好.

乐观锁适用于写入比较少的情况下,即冲突真的很少发生的时候,这样可以省去了锁的开销,加大了系统的整个吞吐量。

但如果经常产生冲突,上层应用会不断的进行重试操作,这样反倒是降低了性能,所以这种情况下用悲观锁就比较合适.

mysql触发器行锁_MySQL 之 视图、触发器、存储过程、函数、事物与数据库锁相关推荐

  1. MySQL 之 视图、触发器、存储过程、函数、事物与数据库锁

    浏览目录: 1.视图 2.触发器 3.存储过程 4.函数 5.事物 6.数据库锁 7.数据库备份 1.视图 视图:是一个虚拟表,其内容由查询定义.同真实的表一样,视图包含一系列带有名称的列和行数据 视 ...

  2. Mysql(三)索引、视图、存储过程、触发器、分区表

    文章目录 一.索引 1.1 索引概述 1.2 索引的基本原理 1.3 索引的优缺点 1.4 索引的创建与删除 1.5 索引分类[逻辑角度] 1.5.1 主键索引 1.5.2 唯一索引 1.5.3 普通 ...

  3. mysql排序行号_mysql 取得行号后再排序

    一.理论准备 Map是键值对的集合接口,它的实现类主要包括:HashMap,TreeMap,Hashtable以及LinkedHashMap等. TreeMap:基于红黑树(Red-Black tre ...

  4. mysql 命令行访问_Mysql 命令行模式访问操作mysql数据库操作

    使用环境 在cmd模式下输入 mysql --version (查看mysql安装的版本). 完整的命令可以通过mysql --help来获取. 本测试使用的Mysql版本是mysql5, 本测试使用 ...

  5. mysql命令行语句_MySql命令行命令和SQL语句

    一.常用mysql命令行命令 1.启动MYSQL服务 net start mysql 停止MYSQL服务 net stop mysql 2.netstat -na|findstr 3306 查看被监听 ...

  6. mysql 命令行 主从复制_MySQL 的主从复制(高级篇)

    首先要明白为什么要用 mysql 的主从复制: 1–在从服务器可以执行查询工作 (即我们常说的读功能),降低主服务器压力:(主库写,从库读,降压) 2–在从主服务器进行备份,避免备份期间影响主服务器服 ...

  7. mysql 分组行号_mysql 显示行号,以及分组排序

    建表: CREATE TABLE `my_tb` ( `id` int(11) NOT NULL AUTO_INCREMENT, `parent_code` varchar(255) DEFAULT ...

  8. mysql采用 级触发_Mysql高级之触发器(trigger)

    触发器是一类特殊的事务 ,可以监视某种数据操作(insert/update/delete),并触发相关操作(insert/update/delete). 看以下事件: 完成下单与减少库存的逻辑 Ins ...

  9. mysql 触发器 库存管理_Mysql中的触发器(库存、用户订单中用到)

    什么是触发器? 触发器是数据库的一个程序,他是用来监听着数据表的某个行为,一旦数据表的这个行为发生了,马上执行相应的sql语句 触发器的语法结构: create trigger触发器的名称触发器事件o ...

最新文章

  1. Java Execution Process
  2. mysql的分页查询
  3. SecureCRTSecureFX_HH_x64_7.0.0.326 crt部署项目到服务器
  4. linux强行卸载qt,Linux下卸载QT SDK
  5. Vue笔记-vue3中.en.dev文件及axios.defaults.baseURL的使用
  6. 年底促销海报还没准备好?电商美工看这里
  7. hadoop在ubuntu上的安装流程
  8. 学习笔记12--Apollo开发平台
  9. c语言入门-程序运行的过程
  10. r语言 linux使用教程,R语言初级教程: R编程环境的搭建
  11. 光模块测试CEI-28G-VSR 浅析1---背景基础介绍
  12. Safari插件机制研究(一)
  13. 初探 Redis 客户端 Lettuce:真香!
  14. linux中rcf命名管道,RCF-进程间为C通讯
  15. 【每日新闻】阿里云成中国唯一全面提供IPv6服务 | 区块链技术可以促进数据的共享以改进流程...
  16. 概率论减法公式的证明
  17. SQL递归查询树型分类数据
  18. SOA Presentation - SOA概念介绍
  19. Git上传项目提示Push rejected: Push to origin/dev was rejected解决办法
  20. 文艺范儿的程序猿和攻城狮们

热门文章

  1. nginx coredump 不产生core文件
  2. ubuntu 16.04 安装Caffe GPU版本
  3. Spark _23 _读取parquet文件创建DataFrame(二)
  4. leetcode 636. Exclusive Time of Functions | 636. 函数的独占时间(Stack)
  5. leetcode 581. Shortest Unsorted Continuous Subarray | 581. 最短无序连续子数组(单调栈)
  6. 【MySQL】[ERR] 1273 - Unknown collation: 'utf8mb4_0900_ai_ci'
  7. 【Java基础】static初始化块
  8. spring注解式参数校验
  9. ThreadLocal用法详解和原理
  10. Leecode03. 无重复字符的最长子串——Leecode大厂热题100道系列