主要看并发事务中不存在则插入(只有key索引)的阻塞情况。

表定义:

mysql> desc user;
+-------------+------------------+------+-----+-------------------+----------------+
| Field | Type | Null | Key | Default | Extra |
+-------------+------------------+------+-----+-------------------+----------------+
| id | int(10) unsigned | NO | PRI | NULL | auto_increment |
| name | varchar(50) | NO | MUL | NULL | |
| password | char(20) | NO | | NULL | |
| regist_time | timestamp | NO | | CURRENT_TIMESTAMP | |
+-------------+------------------+------+-----+-------------------+----------------+
4 rows in set (0.00 sec)

事务隔离级别:RR

mysql版本:5.7

client1:

mysql> begin;
Query OK, 0 rows affected (0.00 sec) mysql> select * from user;
+----+------+----------+---------------------+
| id | name | password | regist_time |
+----+------+----------+---------------------+
| 1 | a | a | 2018-03-11 16:32:43 |
| 2 | b | b | 2018-03-11 16:33:09 |
| 3 | c | c | 2018-03-11 16:33:39 |
+----+------+----------+---------------------+
3 rows in set (0.00 sec) mysql> insert into user(name,password) select 'd','d' from dual where not exist (select name from user where name='d');
Query OK, 1 row affected (0.00 sec)
Records: 1  Duplicates: 0  Warnings: 0 mysql> select * from user;
+----+------+----------+---------------------+
| id | name | password | regist_time |
+----+------+----------+---------------------+
| 1 | a | a | 2018-03-11 16:32:43 |
| 2 | b | b | 2018-03-11 16:33:09 |
| 3 | c | c | 2018-03-11 16:33:39 |
| 4 | d | d | 2018-03-11 17:03:35 |
+----+------+----------+---------------------+
4 rows in set (0.00 sec)

然后启动client2:

mysql> begin;
Query OK, 0 rows affected (0.00 sec) mysql> select * from user;
+----+------+----------+---------------------+
| id | name | password | regist_time |
+----+------+----------+---------------------+
| 1 | a | a | 2018-03-11 16:32:43 |
| 2 | b | b | 2018-03-11 16:33:09 |
| 3 | c | c | 2018-03-11 16:33:39 |
+----+------+----------+---------------------+
3 rows in set (0.00 sec) mysql> select * from user where name='d';
Empty set (0.02 sec) mysql> insert into user (name,password) select 'd','d' from dual where not exists (select name from user where name='d');

client2 执行“ insert into user (name,password) select 'd','d' from dual where not exists (select name from user where name='d'); ”出现阻塞,直到超时或client1 commit。

client2 直接执行插入操作则不会阻塞:

mysql> insert into user(name, password) values ('d','d');
Query OK, 1 row affected (0.00 sec)

client2 执行:

mysql> insert into user (name,password) select 'e','e' from dual where not exists (select name from user where name='e');

也会出现阻塞。但是执行:

mysql> insert into user (name,password) select '12','12' from dual where not exists (select name from user where name='12');
Query OK, 1 row affected (0.02 sec)
Records: 1 Duplicates: 0 Warnings: 0

并不会阻塞。

另:如果已经存在name='d'的数据,client1执行"insert not exists"后并不会插入也不会加锁,client2执行时也不会阻塞。

查看锁(client2 插入'd'时的情况):

mysql> select * from information_schema.innodb_locks;
+-------------------------+-----------------+-----------+-----------+----------------+------------+------------+-----------+----------+-----------+
| lock_id | lock_trx_id | lock_mode | lock_type | lock_table | lock_index | lock_space | lock_page | lock_rec | lock_data |
+-------------------------+-----------------+-----------+-----------+----------------+------------+------------+-----------+----------+-----------+
| 422016582501824:462:4:8 | 422016582501824 | S | RECORD | `test1`.`user` | name | 462 | 4 | 8 | 'd', 11 |
| 162094:462:4:8 | 162094 | X | RECORD | `test1`.`user` | name | 462 | 4 | 8 | 'd', 11 |
+-------------------------+-----------------+-----------+-----------+----------------+------------+------------+-----------+----------+-----------+
2 rows in set, 1 warning (0.02 sec)

client2 当插入'z'时也会阻塞,但lock_data还会是:

mysql> select * from information_schema.innodb_locks;
+----------------+-------------+-----------+-----------+----------------+------------+------------+-----------+----------+------------------------+
| lock_id | lock_trx_id | lock_mode | lock_type | lock_table | lock_index | lock_space | lock_page | lock_rec | lock_data |
+----------------+-------------+-----------+-----------+----------------+------------+------------+-----------+----------+------------------------+
| 162131:462:4:1 | 162131 | X | RECORD | `test1`.`user` | name | 462 | 4 | 1 | supremum pseudo-record |
| 162094:462:4:1 | 162094 | S | RECORD | `test1`.`user` | name | 462 | 4 | 1 | supremum pseudo-record |
+----------------+-------------+-----------+-----------+----------------+------------+------------+-----------+----------+------------------------+
2 rows in set, 1 warning (0.00 sec)

也就是'z'是加锁的上界,插入'x'和'~'也是这种情况。

之所以'12'不会锁,'d'和其以后的都会锁,是因为mysql为了防止幻读,还锁住了下一行,因为最大的是'd',所以锁住区域为('d', +∞),另一个区域是('c', 'd')。如果插入的不是这个区域的都不会阻塞。

RC和RR加锁区别请见:RR和RC复合语句加锁

当client2 插入'A'、'B'时居然不阻塞也插入不了:

mysql> insert into user (name,password) select 'A','A' from dual where not exists (select name from user where name='A');
Query OK, 0 rows affected (0.00 sec)
Records: 0 Duplicates: 0 Warnings: 0 mysql> insert into user (name,password) select 'B','B' from dual where not exists (select name from user where name='B');
Query OK, 0 rows affected (0.00 sec)
Records: 0 Duplicates: 0 Warnings: 0

client1也插入不了'A',只有直接执行时才可以:

mysql> insert into user (name,password) select 'A','A' from dual where not exists (select name from user where name='A');
Query OK, 0 rows affected (0.00 sec)
Records: 0 Duplicates: 0 Warnings: 0 mysql> insert into user(name,password) values('A','A');
Query OK, 1 row affected (0.00 sec)

之所以出现无法插入'A'、'B',是因为不区分大小写,测试一下便知:

mysql> select * from user where name='a';
+----+------+----------+---------------------+
| id | name | password | regist_time |
+----+------+----------+---------------------+
| 1 | a | a | 2018-03-11 16:32:43 |
| 44 | A | A | 2018-03-11 20:56:42 |
+----+------+----------+---------------------+
2 rows in set (0.00 sec)

要想区分大小写,建表时需要相应设置,也可以在查询时使用:

mysql> select * from user where binary name='a';
+----+------+----------+---------------------+
| id | name | password | regist_time |
+----+------+----------+---------------------+
| 1 | a | a | 2018-03-11 16:32:43 |
+----+------+----------+---------------------+
1 row in set (0.01 sec)

另:on duplicate key只适用于unique key,如果不是unique,总是会插入

mysql> insert into user(name,password) values('d','d') on duplicate key update password='e';

这时会插入一条name='d',password='d'的记录。

mysql RR下不存在则插入的更多相关文章

  1. mysql RC下不存在则插入

    mysql版本:5.7 目的:在RC下,name列上仅有key索引,并发插入name时不出现重复数据 RC不加gap lock,并且复合select语句是不加锁的快照读,导致两个事务同时进行都可插入, ...

  2. 我的MYSQL学习心得(八) 插入 更新 删除

    我的MYSQL学习心得(八) 插入 更新 删除 我的MYSQL学习心得(一) 简单语法 我的MYSQL学习心得(二) 数据类型宽度 我的MYSQL学习心得(三) 查看字段长度 我的MYSQL学习心得( ...

  3. Mysql数据库实践操作之————批量插入数据(100万级别的数据)

    第一种方法:使用insert into 插入 从Redis每次获取100条数据,根据条件去插入到Mysql数据库中: 条件: 如果当前队列中的值大于1000条,则会自动的条用该方法,该方法每次获取从队 ...

  4. MySQL RR隔离 读一致性

    MySQL RR 模式下 事务隔离问题: Session 1: mysql> select * from test; +------+------+ | id | name | +------+ ...

  5. mysql三种带事务批量插入

    原文:mysql三种带事务批量插入 c#之mysql三种带事务批量插入 前言 对于像我这样的业务程序员开发一些表单内容是家常便饭的事情,说道表单 我们都避免不了多行内容的提交,多行内容保存,自然要用到 ...

  6. python+mysql:实现一千万条数据插入数据库

    作业要求 构建一个关系模式和课本中的关系movies(title,year,length,movietype,studioname,producerC)一样的关系,名称自定,在这个关系中插入1000万 ...

  7. 如何在mysql客户端即mysql提示符下执行操作系统命令

    环境描述: mysql版本:5.5.57-log 操作系统版本:Red Hat Enterprise Linux Server release 6.6 (Santiago) 需求描述: 在mysql的 ...

  8. 关于mysql安装后在客户端cmd插入语句无法执行的问题

    关于mysql安装后在客户端cmd插入语句无法执行的问题 因为windows cmd默认字符集是gbk,当character_set_client=utf8时,cmd中出现中文会报错:characte ...

  9. Mysql InnoDB下的两种行锁

    今天例举2种常见的Mysql InnoDB下的行锁 现有表dr_test(id pk, name) 数据是 1 zhangsan2 lisi3 wangwu 例子1 事务1 update dr_tes ...

随机推荐

  1. 动态载入DLL所需要的三个函数详解(LoadLibrary,GetProcAddress,FreeLibrary)

    动态载入 DLL 动态载入方式是指在编译之前并不知道将会调用哪些 DLL 函数, 完全是在运行过程中根据需要决定应调用哪些函数. 方法是:用 LoadLibrary 函数加载动态链接库到内存,用 Ge ...

  2. Android-fragment的替换

    fragment的替换:是指一个Activity加载多个Fragment,当某些动作的时候在Activity替换Fragment显示: 昨天写的这几篇博客,Android-fragment简介-fra ...

  3. Sharepoint/Project Server 看不到“安全性”菜单以及子菜单

    在Sharepoint/Project Server 构建后,左侧看不到看不到“服务器设置”菜单,在设置菜单后左侧出现“服务器设置”菜单,但是依然在右侧看不到“安全性”菜单以及子菜单. (这个图是借的 ...

  4. Google Summer of Code礼包

    这个暑假参加google summer of code, 给Google的分布式容器管理系统kubernates开发新的特性,希望从中学习更多的分布式的技术,锻炼自己的编程技巧. 中午在学校的图书馆吗 ...

  5. java 发架包

    // 完整发布流程 clean compile deploy install -U -DskipTests // 把架包安装到本地,跳过测试 install -Dmaven.test.skip=tru ...

  6. ie下警告console未定义

    低版本IE6/7/8/9浏览器没有定义console对象,所以代码会中断执行.自己测试,ie11也没有(打开控制台的情况下可以用) 可以用如下代码完美解决. window.console = wind ...

  7. asp.net Ibatis.net 批量插入数据ORACLE

    在开发中我们有时会遇到需要批量插入数据,最普通的就是每次 插入一条.但是当数据量大道一定的地步会很影响性能.下面例子示范了ibatis.net批量插入 ibatis.net 的XML文件里面使用ite ...

  8. ASP.NET Core 2 学习笔记(四)依赖注入

    ASP.NET Core使用了大量的依赖注入(Dependency Injection, DI),把控制反转(Inversion Of Control, IoC)运用的相当巧妙.DI可算是ASP.NE ...

  9. RTOS双向链表数据结构

    在学习RTOS操作系统时,在任务优先级设置时用到了双向链表,说实话数据结构的东西只是停留在大学上课阶段,并未实践过,在操作系统中看得云里雾里,遂将其单独拿来了进行了一下思考,经过一个上午的摸索逐渐领会 ...

  10. Django路由配置系统、视图函数

    一.路由配置系统(URLconf) URL配置(URLconf)就像Django 所支撑网站的目录.它的本质是URL与要为该URL调用的视图函数之间的映射表:你就是以这种方式告诉Django,对于这个 ...