mysql RR下不存在则插入
主要看并发事务中不存在则插入(只有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下不存在则插入的更多相关文章
- mysql RC下不存在则插入
mysql版本:5.7 目的:在RC下,name列上仅有key索引,并发插入name时不出现重复数据 RC不加gap lock,并且复合select语句是不加锁的快照读,导致两个事务同时进行都可插入, ...
- 我的MYSQL学习心得(八) 插入 更新 删除
我的MYSQL学习心得(八) 插入 更新 删除 我的MYSQL学习心得(一) 简单语法 我的MYSQL学习心得(二) 数据类型宽度 我的MYSQL学习心得(三) 查看字段长度 我的MYSQL学习心得( ...
- Mysql数据库实践操作之————批量插入数据(100万级别的数据)
第一种方法:使用insert into 插入 从Redis每次获取100条数据,根据条件去插入到Mysql数据库中: 条件: 如果当前队列中的值大于1000条,则会自动的条用该方法,该方法每次获取从队 ...
- MySQL RR隔离 读一致性
MySQL RR 模式下 事务隔离问题: Session 1: mysql> select * from test; +------+------+ | id | name | +------+ ...
- mysql三种带事务批量插入
原文:mysql三种带事务批量插入 c#之mysql三种带事务批量插入 前言 对于像我这样的业务程序员开发一些表单内容是家常便饭的事情,说道表单 我们都避免不了多行内容的提交,多行内容保存,自然要用到 ...
- python+mysql:实现一千万条数据插入数据库
作业要求 构建一个关系模式和课本中的关系movies(title,year,length,movietype,studioname,producerC)一样的关系,名称自定,在这个关系中插入1000万 ...
- 如何在mysql客户端即mysql提示符下执行操作系统命令
环境描述: mysql版本:5.5.57-log 操作系统版本:Red Hat Enterprise Linux Server release 6.6 (Santiago) 需求描述: 在mysql的 ...
- 关于mysql安装后在客户端cmd插入语句无法执行的问题
关于mysql安装后在客户端cmd插入语句无法执行的问题 因为windows cmd默认字符集是gbk,当character_set_client=utf8时,cmd中出现中文会报错:characte ...
- Mysql InnoDB下的两种行锁
今天例举2种常见的Mysql InnoDB下的行锁 现有表dr_test(id pk, name) 数据是 1 zhangsan2 lisi3 wangwu 例子1 事务1 update dr_tes ...
随机推荐
- Android-自定义ListView下拉刷新与上拉加载
效果图: 第一步:编写需要在ListView中增加头加载的布局文件,与底部加载的布局文件: 头布局文件: <?xml version="1.0" encoding=" ...
- MySQL Route负载均衡与读写分离Docker环境使用
Docker环境描述 主机名 部署服务 备注 MySQL Route MySQL Route 部署在宿主机上的MySQL Route服务 MySQL Master1 MySQL 5.7.16 Dock ...
- LeetCode138:Copy List with Random Pointer
题目: A linked list is given such that each node contains an additional random pointer which could poi ...
- C#不用union,而是有更好的方式实现
用过C/C++的人都知道有个union,特别好用,似乎char数组到short,int,float等的转换无所不能,也确实是能,并且用起来十分方便.那C#为什么没有这个关键字呢?怎么实现这个功能?其实 ...
- VS2017 无法使用"XXX"附加到应用程序
可能是启用了腾讯的网游,可以关闭游戏,再试一下,如果还是不行,重启一下就可以了.好像是游戏的什么防篡改的作用
- ueditor图片上传插件的使用
在项目里使用到ueditor图片上传插件,以前图片上传都是直接使用js代码直接上传图片,比较麻烦,而且效率也比较低,而ueditor这款插件完美的解决了这个问题,这个是百度开发的一款富文本编辑器,在这 ...
- c# 1-2+3-4.....求和
找规律: 下界:1 上界:n class Program { static void Main(string[] args) { ; ; i <=; i++) { ==) { sum -= i; ...
- hdu5833----高斯消元
题目大意: 给你n个整数,从中选一些数,他们的乘积为一个完全平方数 问有多少种这样的方式,已知这些数的素因素不超过2000. 思路: 一个完全平方数素因素的个数肯定是偶数个. 我们只要从n个数中选取所 ...
- [JavaScript] 根据字符串宽度截取字符串
/** * 根据字符串宽度截取字符串 * @param desc 原始字符串 * @param width 该显示的宽度 * @param fontsize 字体大小 12px * @returns ...
- 【8】JMicro微服务-JMicro ZKUI
ZKUI是一个开源项目,是一个查看,修改ZK数据非常方便的工具.JMicro基于ZK做服务治理,配置管理,因此使用ZKUI会提供非常大的方便. Github地址:https://github.com/ ...