icode icode
首页
  • Android学习

    • 📁基础内容
    • 📺AndroidCore
    • 🎨Android-UI
    • 🏖️Components
    • 📊Fragment
    • 🔗网络操作
    • 🔏异步机制
    • 📦数据存储
    • 🗃️Gradle
  • 学习笔记

    • 『框架』笔记
    • 『Kotlin』笔记
    • 《Vue》笔记
    • 《Git》学习笔记
    • 『Bug踩坑记录』
  • ListView
  • RecyclerView
  • ViewPager
  • Java笔记

    • 🟠JavaSE
    • 🟢JavaWeb
    • 🔴JavaEE
    • ⚪JavaTopic
    • 🍳设计模式
  • 计算机基础

    • 📌计算机网络
    • 🔍数据结构
    • 📦数据库
    • 💻OS
  • 技术文档
  • GitHub技巧
  • Nodejs
  • 博客搭建
  • 学习
  • 面试
  • 心情杂货
  • 实用技巧
  • 友情链接
  • 关于

    • 📫关于我
  • 收藏

    • 网站
    • 资源
    • Vue资源
  • 分类
  • 标签
  • 归档
GitHub (opens new window)

iqqcode

保持对技术的探索实践与热爱
首页
  • Android学习

    • 📁基础内容
    • 📺AndroidCore
    • 🎨Android-UI
    • 🏖️Components
    • 📊Fragment
    • 🔗网络操作
    • 🔏异步机制
    • 📦数据存储
    • 🗃️Gradle
  • 学习笔记

    • 『框架』笔记
    • 『Kotlin』笔记
    • 《Vue》笔记
    • 《Git》学习笔记
    • 『Bug踩坑记录』
  • ListView
  • RecyclerView
  • ViewPager
  • Java笔记

    • 🟠JavaSE
    • 🟢JavaWeb
    • 🔴JavaEE
    • ⚪JavaTopic
    • 🍳设计模式
  • 计算机基础

    • 📌计算机网络
    • 🔍数据结构
    • 📦数据库
    • 💻OS
  • 技术文档
  • GitHub技巧
  • Nodejs
  • 博客搭建
  • 学习
  • 面试
  • 心情杂货
  • 实用技巧
  • 友情链接
  • 关于

    • 📫关于我
  • 收藏

    • 网站
    • 资源
    • Vue资源
  • 分类
  • 标签
  • 归档
GitHub (opens new window)
  • MySQL

    • DQL

    • MySQL进阶

      • 事务
      • B+树
      • 索引概念篇
      • 索引定义
      • 回表&索引覆盖&最左
      • 索引语法
      • 慢SQL优化
      • 索引失效及解决
      • 模糊查询优化
      • SQL语句执行流程
      • 存储引擎
      • MySQL锁
        • 【锁的实现】
        • 1. Mysql锁分类
        • 2. MyISAM 表锁-偏读
          • 如何加表锁
        • 3. InnoDB 行锁-偏写
          • 行锁介绍
          • InnoDB 的行锁模式
        • 4. 索引行锁升级为表锁
          • 总结
      • 主从复制
      • MVCC机制
      • 行锁、表锁
  • 数据库
  • MySQL
  • MySQL进阶
iqqcode
2021-06-17
目录

MySQL锁

# 【锁的实现】

InnoDB行锁:只有通过索引条件检索数据,InnoDB才会使用行级锁,否则,InnoDB将使用表锁!

InnoDB-行锁是基于索引来实现的:

  • 行锁必须有索引才能实现,否则会自动锁全表

  • 两个事务不能锁同一个索引

  • 类型转换导致行锁升级为表锁。在MySql的写语句中,给表列赋值与表类型不符合时,MySql底层的优化器发挥作用,会做一个强制类型转化,此时能正常操作,但会导致行锁升级为表锁


# 1. Mysql锁分类

锁概述锁是计算机协调多个进程或线程并发访问某一资源的机制(避免资源争抢)。

**从对数据操作的粒度分 : **

  • 表锁:操作时,会锁定整个表

  • 行锁:操作时,会锁定当前操作行

从对数据操作的类型分:

  • 读锁(共享锁):读锁会阻塞写,但是不会阻塞读
  • 写锁(排它锁):不能与其他锁并存。当前操作没有完成之前,它会阻断其他写锁和读锁

lock table user read;   -- 加读锁

lock table user write;  -- 加写锁

unlock tables;          -- 释放锁
1
2
3
4
5

image-20201008111141771

读锁: session1 对 TableA 加读锁

  • session1可以读自己加锁的 TableA,不能改自己的TableA,不可以读其他表
  • session2 可以读TableA,改TableA时会阻塞等待,持有者释放锁之后才能改 ,可以读其他表

写锁: session1 对 TableA 加写锁

  • session1可以读自己加锁的 TableA,可以改自己的TableA,不可以读其他表
  • session2 读TableA时会阻塞等待,持有者释放锁之后才能可以读。不可以改TableA,可以读其他表

各存储引擎对锁的支持情况:

存储引擎 表级锁 行级锁 页面锁
MyISAM 支持 不支持 不支持
InnoDB 支持 支持 不支持

MySQL这3种锁的特性可大致归纳如下 :

image-20200824163558057

从上述特点可见,很难笼统地说哪种锁更好,只能就具体应用的特点来说哪种锁更合适!

仅从锁的角度来说:表级锁更适合于以查询为主,只有少量按索引条件更新数据的应用,如Web 应用;

行级锁则更适合于有大量按索引条件并发更新少量不同数据,同时又有并查询的应用,如一些在线事务处理(OLTP)系统。

# 2. MyISAM 表锁-偏读

MyISAM 存储引擎只支持表锁,这也是MySQL开始几个版本中唯一支持的锁类型。

# 如何加表锁

加锁的过程是自动的。MyISAM 在执行查询语句(SELECT)前,会自动给涉及的所有表加读锁,在执行更新操作(UPDATE、DELETE、INSERT 等)前,会自动给涉及的表加写锁,这个过程并不需要用户干预,因此,用户一般不需要直接用 LOCK TABLE 命令给 MyISAM 表显式加锁。

显示加表锁语法:

加读锁 : lock table table_name read;

加写锁 : lock table table_name write;
1
2
3

锁模式的相互兼容性如表中所示:

1553905621992

由上表可见:

  • 简而言之,就是读锁会阻塞写,但是不会阻塞读。

  • 而写锁,则既会阻塞读,又会阻塞写。

此外,MyISAM 的读写锁调度是写优先,这也是MyISAM不适合做写为主的表的存储引擎的原因。因为写锁后,其他线程不能做任何操作,大量的更新会使查询很难得到锁,从而造成永远阻塞。


# 3. InnoDB 行锁-偏写

# 行锁介绍

行锁特点 :

  • 偏向InnoDB 存储引擎,开销大,加锁慢;
  • 会出现死锁;
  • 锁定粒度最小,发生锁冲突的概率最低,并发度也最高。

InnoDB 与 MyISAM 的最大不同有两点:一是支持事务;二是采用了行锁


# InnoDB 的行锁模式

InnoDB 支持行锁和表锁

  • 对于UPDATE、DELETE和INSERT语句,InnoDB会自动给涉及数据集加排他锁(X);

  • 对于普通SELECT语句,InnoDB不会加任何锁;

可以通过以下语句显示给记录集加共享锁或排他锁

共享锁(S): SELECT * FROM table_name WHERE ... LOCK IN SHARE MODE

排他锁(X): SELECT * FROM table_name WHERE ... FOR UPDATE
1
2
3

# 4. 索引行锁升级为表锁

如果不通过索引条件检索数据,那么InnoDB将对表中的所有记录加锁,实际效果跟表锁一样。

**索引失效,最终行锁变为表锁 **

为何索引会失效?——— 使用的索引字段的数据类型 与 定义的数据类型不符。

查看当前表的索引 :

show  index  from test_innodb_lock ; 
1

由于执行更新时 , name字段本来为varchar类型, 我们是作为int类型使用,存在类型转换的异常,所以索引失效,行锁变为表锁

# 总结

优化建议:

  • 避免无索引行锁升级为表锁,尽可能让所有数据检索都能通过索引来完成
  • 合理设计索引,尽量缩小锁的范围
  • 尽可能减少索引条件,及索引范围,避免间隙锁
  • 尽量控制事务大小,减少锁定资源量和时间长度
编辑 (opens new window)
上次更新: 2021/06/27, 10:49:09
存储引擎
主从复制

← 存储引擎 主从复制→

最近更新
01
匿名内部类
10-08
02
函数式接口
10-08
03
ARouter-Kotlin踩坑
10-05
更多文章>
Theme by Vdoing | Copyright © 2021-2023 iqqcode | MIT License | 备案号-京ICP备2021028793号
  • 跟随系统
  • 浅色模式
  • 深色模式
  • 阅读模式
×