一.概述

  • 读-读:并发不存在问题,不需要加锁
  • 写-写:并发存在问题,可能会造成脏写(一个事务没有写完,另一个事务也对相同的数据进行写),但是这种情况,任何一种隔离级别都不允许发生,在隔离级别的时候就解决了。
  • 读-写/写-读:会造成脏读,幻读,不可重复读的问题。每个数据厂商对它的支持也是不相同的
    • 解决方案:

      • 方案一:读操作利用MVCC,写进行加锁。
      • 方案二:读写都加锁。

怎么加锁:数据库自己就进行加锁,不需要手动加锁。除非想演示效果,就自己开事务自己加锁

  • 表锁:

    • lock tables 表名 read      //表级读锁
    • lock tables 表名 write     //表级写锁
    • unlock //释放锁
  • 行锁:
    • SQL语句 for share         //行级读锁
    • SQL语句 for update       //行级写锁
    • commit //释放锁

二.锁

基于锁的属性分类:

  • 共享锁(S):读锁,可以并发的读数据
  • 排他锁(X):写锁,独占锁

基于锁的粒度分类:(MyISAM使用表锁,innodb一般使用行锁)

  • 表级锁

    • 表级读锁:在执行select 之前,会给涉及的所有表加读锁。
    • 表级写锁:在执行增删改之前,会给涉及的所有表加写锁。
    • 意向锁:就是一个标志,某一行加上锁之后,就给整个表加一个意向锁,有新的锁加入表锁 ,就需要等待获得意向锁。
    • 自增锁:就是表中的主键自增(auto-increment)也需要加锁。
    • 元数据锁(MDL):就是对表进行DDL的时候,需要进行加锁。
  • 行级锁
    • 记录锁:锁住每一条记录,就是行级读锁/行级写锁。
    • 间隙锁(Gap):就是为了解决幻读而提出的。(幻读除了使用间隙锁解决,还可以使用MVCC解决),间隙锁还可能造成死锁。

    • 临界锁:是记录锁和gap锁的合体,即锁住了某条记录,又阻止在该记录前插入新纪录。
    • 插入意向锁:是一种间隙锁,专门针对的是数据行的插入操作,多个事务插入相同的索引间隙时,只要不是插入到相同的位置,则不需要进行锁等待。插入意向锁锁定了索引之间的间隙,但是插入意向锁之间没有互相阻塞。(间隙锁让范围之间不能插入,插入意向锁虽然也锁住了一个范围,但是不是同一个位置的就可以插入) 
  • 页锁:锁着一页,是介于表锁和行锁之间,并发度一般。会出现死锁。

加锁方式分类

  • 隐式锁:自动添加的,在insert的时候就会添加一个隐式锁。比如事务A在insert一条数据后,还没有提交,此时另一个事务B要修改这条新添加的数据,就不会成功。

  • 显示锁:就是我们上面学的锁

其他的锁

  • 死锁:两个或者多个事务在同一资源上相互占用,并请求锁定对方占用的资源,从而导致恶性循环。

    • 导致死锁的场景:

      • 间隙锁导致的死锁:比如两个事务A,B,都锁了(3,8)之间的范围,事务A进行插入数据到(3,8)发生阻塞,同理事务B进行插入数据到(3,8)也发生阻塞。但是它俩都僵持着,互相进行阻塞,就发生死锁
      • 页锁导致的死锁:事务A,B。A先锁住页1然后锁了页2,B先锁了页2然后锁了页1。
    • 出现死锁,有2种策略
      • 方式一:直接进入等待,直到超时。这个超时时间可以通过innodb_lock_wait_timeout来设置。
      • 方式二:死锁检测,发现死锁,主动回滚死锁链中的某一个事务,让其他事务继续执行。通过innodb_deadlock_detect设置为on来开启。
    • 如何避免出现死锁:(了解)
      • 如果不同的程序并发存取多个表,尽量以相同的顺序访问表。
      • 在程序以批量方式处理数据的时候,如果已经对数据排序,尽量保证每个线程按照固定的顺序来处理记录。
      • 在事务中,如果需要更新记录,应直接申请足够级别的排他锁,而不应该先申请共享锁,更新时在申请排他锁,因为在当前用户申请排他锁时,其他事务可能已经获得了相同记录的共享锁,从而造成锁冲突或者死锁。
      • 尽量使用较低的隔离级别。
      • 尽量使用索引访问数据,使加锁更加准确,从而减少锁冲突的机会。
      • 合理选择事务的大小,小事务发生锁冲突的概率更低。
      • 尽量用相等的条件访问数据,可以避免Next-Key锁对并发插入的影响。
      • 不要申请超过实际需要的锁级别,查询时尽量不要显示加锁。
      • 对于一些特定的事务,可以表锁来提高处理速度或减少死锁的概率。
  • 全局锁:让整个数据库实例都加锁,不管是DDL还是DML都加锁。
    • 使用场景:全库逻辑备份。
    • 命令Flush tables with read lock;开启

三.MVCC

MVCC:多版本并发控制:多版本:undo链实现;控制:readView。主要针对于RC和RR两种事务。

为了提高数据库并发性能,用更好的方式去处理读-写冲突。读就是采用快照读,写就是采用当前读(加锁)。
前提知识:

  • 快照读:不加锁的非阻塞读,比如select * from ...
  • 当前读:读取时还要保证其他并发事务不能修改当前记录,会对读取的记录进行加锁。不仅限于读数据,修改数据也属于当前读。
  • 行格式的隐藏列

  • undo 版本链:每一个数据行的隐藏指针指向自己undo版本链,undo版本链中应该存储的之前的逆操作,但是为了好理解,下图就画成数据原来的样子

  • ReadView:就是每一个事务在使用MVCC机制进行快照读操作时产生的读视图。每一个事务都会产生一个ReadView。

MVCC的实现依赖于隐藏字段,undo Log,ReadView

说白了,就是事务生成的ReadView与要修改的这条记录的隐藏列的tri_id进行比较,看是读取undo链中的历史呢,还是读取当前最新的数据呢。

MVCC如何解决幻读?

  • A事务先进行读,然后B事务中途插入一条数据后提交。A事务再次读,根据MVCC,发现trx_id大于low_limit_id,说明实在readView生成之后,才出现的事务B,所以不能读取该数据。就不会出现幻读。

寄语:生活像一只蝴蝶,没有破茧的勇气,哪来飞舞的美丽

MySQL读写问题(锁)的更多相关文章

  1. mysql 行级锁的使用以及死锁的预防

    一.前言 mysql的InnoDB,支持事务和行级锁,可以使用行锁来处理用户提现等业务.使用mysql锁的时候有时候会出现死锁,要做好死锁的预防. 二.MySQL行级锁 行级锁又分共享锁和排他锁. 共 ...

  2. mysql事务和锁InnoDB

    背景 MySQL/InnoDB的加锁分析,一直是一个比较困难的话题.我在工作过程中,经常会有同事咨询这方面的问题.同时,微博上也经常会收到MySQL锁相关的私信,让我帮助解决一些死锁的问题.本文,准备 ...

  3. MySQL中的锁(表锁、行锁)

    锁是计算机协调多个进程或纯线程并发访问某一资源的机制.在数据库中,除传统的计算资源(CPU.RAM.I/O)的争用以外,数据也是一种供许多用户共享的资源.如何保证数据并发访问的一致性.有效性是所在有数 ...

  4. mysql事务和锁InnoDB(转)

    背景 MySQL/InnoDB的加锁分析,一直是一个比较困难的话题.我在工作过程中,经常会有同事咨询这方面的问题.同时,微博上也经常会收到MySQL锁相关的私信,让我帮助解决一些死锁的问题.本文,准备 ...

  5. MySQL系列(五)---总结MySQL中的锁

    MySQL中的锁 目录 MySQL系列(一):基础知识大总结 MySQL系列(二):MySQL事务 MySQL系列(三):索引 MySQL系列(四):引擎 概述 MyISAM支持表锁,InnoDB支持 ...

  6. MySQL事务与锁

    MySQL事务与锁 锁的基本概念 锁是计算机协调多个进程或线程并发访问某一资源的机制. 相对其他数据库而言,MySQL的锁机制比较简单,其最显著的特点是不同的存储引擎支持不同的锁机制.比如,MyISA ...

  7. Mysql中的锁机制

    原文:http://blog.csdn.net/soonfly/article/details/70238902 锁是计算机协调多个进程或线程并发访问某一资源的机制.在数据库中,除传统的 计算资源(如 ...

  8. Mysql事务及锁

    一.事务(Transaction)及其ACID属性 事务是由一组SQL语句组成的逻辑处理单元,事务具有以下4个属性,通常简称为事务的ACID属性:1.原子性(Atomicity):事务是一个原子操作单 ...

  9. MySQL学习之——锁(行锁、表锁、页锁、乐观锁、悲观锁等)

    转载. https://blog.csdn.net/mysteryhaohao/article/details/51669741 锁,在现实生活中是为我们想要隐藏于外界所使用的一种工具.在计算机中,是 ...

随机推荐

  1. 【九度OJ】题目1191:矩阵最大值 解题报告

    [九度OJ]题目1191:矩阵最大值 解题报告 标签(空格分隔): 九度OJ http://ac.jobdu.com/problem.php?pid=1191 题目描述: 编写一个程序输入一个mXn的 ...

  2. 【机器学*】k*邻算法-02

    k邻*算法具体应用:2-2约会网站配对 心得体会: 1.对所有特征值进行归一化处理:将特征值单位带来的距离影响消除,使所有特征同权重--然后对不同的特征进行加权2.对于相互独立的特征,可以通过建立(特 ...

  3. 第四十五个知识点:描述一些对抗RSA侧信道攻击的基础防御方法

    第四十五个知识点:描述一些对抗RSA侧信道攻击的基础防御方法 原文地址:http://bristolcrypto.blogspot.com/2015/08/52-things-number-45-de ...

  4. 预训练模型时代:告别finetune, 拥抱adapter

    NLP论文解读 原创•作者 |FLIPPED 研究背景 随着计算算力的不断增加,以transformer为主要架构的预训练模型进入了百花齐放的时代.BERT.RoBERTa等模型的提出为NLP相关问题 ...

  5. From Hero to Zero

    题目描述: 有一天,小明给了你两个数字n和k,现在,你需要对数字n进行一下操作: 对于每一步操作,你可以选择下面其中一个项目: 1.将n的值减少1. 2.如果n能被k整除,可以使n/k 比如:n=27 ...

  6. 使用用支付宝时,返回的数据中subject为中文时验签失败

    解决方法为: 来自为知笔记(Wiz)

  7. SQL高级优化系列

    目录 SQL高级优化系列(一)之MySQL优化 SQL高级优化系列(二)之MySQL架构 SQL高级优化系列(三)之存储引擎 SQL高级优化系列(四)之SQL优化 SQL高级优化系列(五)之执行计划 ...

  8. 日志收集系统系列(四)之LogAgent优化

    实现功能 logagent根据etcd的配置创建多个tailtask logagent实现watch新配置 logagent实现新增收集任务 logagent删除新配置中没有的那个任务 logagen ...

  9. Kube-OVN 0.6.0 发布,支持 IPv6、流量镜像及更多功能

    Kube-OVN 是一个基于 OVN 的 Kubernetes 开源网络系统. 本次更新主要包含了以下内容: 1. 支持流量镜像 在安装 Kube-OVN 时可以开启 mirror 选项,会自动在每个 ...

  10. 阿里云服务器ECS Ubuntu16.04 初次使用配置教程(图形界面安装)

    原文链接:? 传送门 前一阵子购买了阿里云的云服务器ECS(学生优惠),折腾了一阵子后对有些东西不太满意,所以就重新初始化了磁盘,刚好要重新安装图形界面,于是就顺手写了这么一篇文章. 第一次登陆服务器 ...