zoukankan      html  css  js  c++  java
  • mysql优化二之锁机制

    mysql优化二之锁机制

    mysql提供了锁机制和MVCC机制来保证并发操作的安全性,这里主要讨论锁机制,
    MVCC见下篇文章

    mysql的锁按照锁粒度可分为行锁与表锁,按照操作类型划分可读锁和写锁
    InnoDB存储引擎支持表锁和行锁,默认锁为行锁,MyIsam只支持表锁
    锁粒度越高则并发性越好

    表锁

    一、操作语法

    1、 show open tables;查看数据库中哪些表加了锁


    in-use为0则表示未加锁

    2、 lock table (table_name) read(write)

    3、 unlock tables;解锁

    二、示例

    1、 读锁
    开了两个mysql客户端,左边客户端中给mytest数据库中的test1表加了读锁

    • 左客户端执行读操作,右客户端执行读操作,可以看到两者都可以执行

    • 对于同一数据库中的其他表操作,左客户端拒绝执行,右客户端可以执行

    • 对于被锁住的表执行写操作


    左客户端拒绝执行

    右客户端阻塞住

    此时我们解锁

    可以看到右客户端立即被释放并正确执行操作


    2、 写锁
    给左客户端test1表加上写锁

    • 左客户端读/写正常,右客户端读/写均被阻塞

    • 左客户端读/写该数据库中的其他表拒绝执行,右客户端可以正常执行


    行锁

    因为mysql默认的存储引擎是InnoDB,而InnoDB默认为行锁,我们要测试行锁首先需要把mysql的自动提交关闭

    • 行级锁读操作两个客户端互不影响

    • 写操作如果操作的不是同一行,也不影响,若操作的是同一行则后一个客户端的请求被阻塞,直到前一个客户端的请求提交


    注意点

    1、 间隙锁

    我们在左边客户端对id在(0,5]范围内的数据做写操作,但是由于数据表中没有id为4的数据行,按理说右边数据库插入一个id为4的数据行不会被锁住,但是事实表明它被锁住了。这既是mysql的间隙锁机制。因此我们在数据库操作时其实应该避免这种间隙的产生,我们可以在表里设置一个状态位,当要删除某一数据行时,可以选择将该状态位设置为无效而不是真正的删除。

    2、 索引失效

    • 当where查询条件没有索引时,行锁变表锁

    • 当where查询条件有索引但是索引失效时,行锁仍然变表锁


    当左客户端在已经减了索引的loc列用int型来查找时,索引失效就会导致整张表被锁住

    行锁变表锁的原因:mysql的行锁是用索引实现的

    3、 如何锁住一行


    在select语句找出某一行之后加一个for update

  • 相关阅读:
    luogu 2627 修剪草坪
    luogu2746 [USACO5.3]校园网Network of Schools
    luogu 1558 色板游戏
    luogu 2827 蚯蚓
    POJ 2559 Largest Rectangle in a Histogram
    luogu 1886 滑动窗口
    luogu 1090 合并果子
    uva 11572
    uva 12626
    uva 10222
  • 原文地址:https://www.cnblogs.com/huanglf714/p/10750520.html
Copyright © 2011-2022 走看看