zoukankan      html  css  js  c++  java
  • mysql 共享锁-排它锁

    转  InnoDB 行级锁

    http://www.cnblogs.com/dongqingswt/archive/2013/03/28/2987367.html

    InnoDB 行级锁

    分类: 数据库

    nnoDB的行锁模式及加锁方法

    InnoDB实现了以下两种类型的行锁。 

     共享锁(S):允许一个事务去读一行,阻止其他事务获得相同数据集的排他锁。

     排他锁(X):允许获得排他锁的事务更新数据,阻止其他事务取得相同数据集的共享读锁和排他写锁。

    另外,为了允许行锁和表锁共存,实现多粒度锁机制,InnoDB还有两种内部使用的意向锁(Intention Locks),这两种意向锁都是表锁。

     意向共享锁(IS):事务打算给数据行加行共享锁,事务在给一个数据行加共享锁前必须先取得该表的IS锁。

     意向排他锁(IX):事务打算给数据行加行排他锁,事务在给一个数据行加排他锁前必须先取得该表的IX锁。

    上述锁模式的兼容情况具体如表20-6所示。

    表20-6     InnoDB行锁模式兼容性列表

    请求锁模式

     

       是否兼容

     

    当前锁模式

    X

    IX

    S

    IS

    X

    冲突

    冲突

    冲突

    冲突

    IX

    冲突

    兼容

    冲突

    兼容

    S

    冲突

    冲突

    兼容

    兼容

    IS

    冲突

    兼容

    兼容

    兼容

    如果一个事务请求的锁模式与当前的锁兼容,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。

    用SELECT ... IN SHARE MODE获得共享锁,主要用在需要数据依存关系时来确认某行记录是否存在,并确保没有人对这个记录进行UPDATE或者DELETE操作。但是如果当前事务也需要对该记录进行更新操作,则很有可能造成死锁,对于锁定行记录后需要进行更新操作的应用,应该使用SELECT... FOR UPDATE方式获得排他锁。

    在如表20-7所示的例子中,使用了SELECT ... IN SHARE MODE加锁后再更新记录,看看会出现什么情况,其中actor表的actor_id字段为主键。

    表20-7           InnoDB存储引擎的共享锁例子

    session_1

    session_2

    mysql> set autocommit = 0;

    Query OK, 0 rows affected (0.00 sec)

    mysql> select actor_id,first_name,last_name from actor where actor_id = 178;

    +----------+------------+-----------+

    | actor_id | first_name | last_name |

    +----------+------------+-----------+

    | 178      | LISA       | MONROE   |

    +----------+------------+-----------+

    1 row in set (0.00 sec)

    mysql> set autocommit = 0;

    Query OK, 0 rows affected (0.00 sec)

    mysql> select actor_id,first_name,last_name from actor where actor_id = 178;

    +----------+------------+-----------+

    | actor_id | first_name | last_name |

    +----------+------------+-----------+

    | 178      | LISA       | MONROE   |

    +----------+------------+-----------+

    1 row in set (0.00 sec)

    当前session对actor_id=178的记录加share mode 的共享锁:

    mysql> select actor_id,first_name,last_name from actor where actor_id = 178 lock in share mode;

    +----------+------------+-----------+

    | actor_id | first_name | last_name |

    +----------+------------+-----------+

    | 178      | LISA       | MONROE   |

    +----------+------------+-----------+

    1 row in set (0.01 sec)

    其他session仍然可以查询记录,并也可以对该记录加share mode的共享锁:

    mysql> select actor_id,first_name,last_name from actor where actor_id = 178 lock in share mode;

    +----------+------------+-----------+

    | actor_id | first_name | last_name |

    +----------+------------+-----------+

    | 178      | LISA       | MONROE   |

    +----------+------------+-----------+

    1 row in set (0.01 sec)

    当前session对锁定的记录进行更新操作,等待锁:

    mysql> update actor set last_name = 'MONROE T' where actor_id = 178;

    等待

    其他session也对该记录进行更新操作,则会导致死锁退出:

    mysql> update actor set last_name = 'MONROE T' where actor_id = 178;

    ERROR 1213 (40001): Deadlock found when trying to get lock; try restarting transaction

    获得锁后,可以成功更新:

    mysql> update actor set last_name = 'MONROE T' where actor_id = 178;

    Query OK, 1 row affected (17.67 sec)

    Rows matched: 1  Changed: 1 Warnings: 0

    当使用SELECT...FOR UPDATE加锁后再更新记录,出现如表20-8所示的情况。

    表20-8             InnoDB存储引擎的排他锁例子

    session_1

    session_2

    mysql> set autocommit = 0;

    Query OK, 0 rows affected (0.00 sec)

    mysql> select actor_id,first_name,last_name from actor where actor_id = 178;

    +----------+------------+-----------+

    | actor_id | first_name | last_name |

    +----------+------------+-----------+

    | 178      | LISA       | MONROE   |

    +----------+------------+-----------+

    1 row in set (0.00 sec)

    mysql> set autocommit = 0;

    Query OK, 0 rows affected (0.00 sec)

    mysql> select actor_id,first_name,last_name from actor where actor_id = 178;

    +----------+------------+-----------+

    | actor_id | first_name | last_name |

    +----------+------------+-----------+

    | 178      | LISA       | MONROE   |

    +----------+------------+-----------+

    1 row in set (0.00 sec)

    当前session对actor_id=178的记录加for update的共享锁:

    mysql> select actor_id,first_name,last_name from actor where actor_id = 178 for update;

    +----------+------------+-----------+

    | actor_id | first_name | last_name |

    +----------+------------+-----------+

    | 178      | LISA       | MONROE   |

    +----------+------------+-----------+

    1 row in set (0.00 sec)

    其他session可以查询该记录,但是不能对该记录加共享锁,会等待获得锁:

    mysql> select actor_id,first_name,last_name from actor where actor_id = 178;

    +----------+------------+-----------+

    | actor_id | first_name | last_name |

    +----------+------------+-----------+

    | 178      | LISA       | MONROE   |

    +----------+------------+-----------+

    1 row in set (0.00 sec)

    mysql> select actor_id,first_name,last_name from actor where actor_id = 178 for update;

    等待

    当前session可以对锁定的记录进行更新操作,更新后释放锁:

    mysql> update actor set last_name = 'MONROE T' where actor_id = 178;

    Query OK, 1 row affected (0.00 sec)

    Rows matched: 1  Changed: 1 Warnings: 0

    mysql> commit;

    Query OK, 0 rows affected (0.01 sec)

    其他session获得锁,得到其他session提交的记录:

    mysql> select actor_id,first_name,last_name from actor where actor_id = 178 for update;

    +----------+------------+-----------+

    | actor_id | first_name | last_name |

    +----------+------------+-----------+

    | 178      | LISA       | MONROE T |

    +----------+------------+-----------+

    1 row in set (9.59 sec)

  • 相关阅读:
    objective-c数组
    objective-c可变数组
    objective-c可变字典
    objective-c字典
    有1、2、3、4个数字,能组成多少个互不相同且无重复数字的三位数?都是多少?
    将一个正整数分解质因数。例如:输入90,打印出90=2*3*3*5
    一球从100米高度自由落下,每次落地后反跳回原高度的一半;再落下,求它在第10次落地时,共经过多少米?
    求123456789-23456789-3456789-456789-...-9的值
    编写一个程序,求s=1+(1+2)+(1+2+3)+…+(1+2+3+…+n)的值
    Unity3D笔记 GUI 二 、实现选项卡一窗口
  • 原文地址:https://www.cnblogs.com/taek/p/4744902.html
Copyright © 2011-2022 走看看