在其他大部分DBMS里都有序列的概念,即Sequence或Generator。

而mysql里没有,但有时真的很有用。下面分别用存储过程和函数来模拟序列,并用程序模拟并发场景来测试原子性和完整性,是否能达到预期。

序列表定义如下:

CREATE TABLE `seq` (
`id` BIGINT(20) NOT NULL,
`busi` VARCHAR(50) NULL DEFAULT NULL,
`val` BIGINT(20) NULL DEFAULT NULL,
`remark` VARCHAR(50) NULL DEFAULT NULL,
PRIMARY KEY (`id`)
)
COLLATE='utf8_general_ci'
ENGINE=InnoDB
;

先把模拟调用程序放这里,因为它是不变的:

            for (int j = ; j < ; j++)
{
Thread t1 = new Thread(() =>
{
for (int i = ; i < ; i++)
{
using (var db = new DbCtxt())
{
//long val = db.Sql("select nextval();").QuerySingle<long>();
long val = db.Sql("call nextval();").QuerySingle<long>();
Console.WriteLine(val+" = "+Thread.CurrentThread.ManagedThreadId);
}
}
});
t1.Start();
}

1、假设存储过程不加事务,读取时不for update

      declare v bigint ;
-- start transaction;
set v = (select val from seq where busi = 'mat');
set v = v + 1;
update seq set val = v where busi = 'mat';
-- commit;
select v;

测试结果:会出现并发读和写数据,现象就是预期序列增加1000,实际每次测试都是增加700~800不定。

2、假设存储过程不加事务,读取时加for update

      declare v bigint ;
-- start transaction;
set v = (select val from seq where busi = 'mat' for update);
set v = v + 1;
update seq set val = v where busi = 'mat';
-- commit;
select v;

测试结果:会出现并发读和写数据,现象就是预期序列增加1000,实际每次测试都是增加700~800不定。

3、假设存储过程加事务,读取时不加for update

      declare v bigint ;
start transaction;
set v = (select val from seq where busi = 'mat' );
set v = v + 1;
update seq set val = v where busi = 'mat';
commit;
select v;

测试结果:每次测试都会出现 Deadlock死锁,并且是很快(val增加不到100)就出现死锁。

4、假设存储过程加事务,读取时也加for update

      declare v bigint ;
start transaction;
set v = (select val from seq where busi = 'mat' for update);
set v = v + 1;
update seq set val = v where busi = 'mat';
commit;
select v;

测试结果:多线程读取,预期序列增加1000,实际每次测试都增加1000,符合预期。

5、用函数模拟,函数不允许显示或隐式的开启事务,先测试读取时不加for update

      declare v bigint ;
-- start transaction;
set v = (select val from seq where busi = 'mat' );
set v = v + 1;
update seq set val = v where busi = 'mat';
-- commit;
return v;

测试结果:多线程读取,预期序列增加1000,实际每次测试都增加1000,符合预期。

6、用函数模拟,测试读取时加for update

      declare v bigint ;
-- start transaction;
set v = (select val from seq where busi = 'mat' for update);
set v = v + 1;
update seq set val = v where busi = 'mat';
-- commit;
return v;

测试结果:多线程读取,预期序列增加1000,实际每次测试都增加1000,符合预期。

总结:若用函数模拟最为简单,不用考虑是否锁定行for update,调用方式 select nextval();

若用存储过程模拟,必须要考虑锁定行for update,且多sql前后要加事务管理,调用方式 call nextval();

mysql 用存储过程和函数分别模拟序列的更多相关文章

  1. MySQL:存储过程和函数

    存储过程和函数 一.创建存储过程和函数 1.创建存储过程 语法: CREATE PROCEDURE sp_name ([proc_parameter[,...]]) [characteristic . ...

  2. MYSQL的存储过程和函数简单写法

    存储过程 MySQL中,创建存储过程的基本形式如下: CREATE PROCEDURE sp_name ([proc_parameter[,...]]) [characteristic ...] ro ...

  3. Mysql中存储过程和函数的写法

    MySQL中,创建存储过程的基本形式如下: CREATE PROCEDURE sp_name ([proc_parameter[,...]]) [characteristic ...] routine ...

  4. MySQL之存储过程和函数

    存储过程和函数: 1.创建存储过程和函数: 存储过程: delimiter $$ create procedure proc_name() BEGIN 查询语句; // 记得加分号 END $$ de ...

  5. mysql的存储过程,函数,事件,权限,触发器,事务,锁,视图,导入导出

    1.创建过程 1.1 简单创建 -- 创建员工表 DROP TABLE IF EXISTS employee; CREATE TABLE employee( id int auto_increment ...

  6. mySQL查看存储过程、函数、视图、触发器

    一.查看存储过程 1.show procedure status; //查看所有的 2.show create procedure proc_AllUser[proc_name]; 查看proc_Al ...

  7. navicat 给mysql 添加存储过程(函数)

    BEGIN DECLARE i INT default 0; DECLARE num int default 0; DECLARE count1 int default 0; DECLARE coun ...

  8. 我的MYSQL学习心得(十) 自定义存储过程和函数

    我的MYSQL学习心得(十) 自定义存储过程和函数 我的MYSQL学习心得(一) 简单语法 我的MYSQL学习心得(二) 数据类型宽度 我的MYSQL学习心得(三) 查看字段长度 我的MYSQL学习心 ...

  9. MySQL 存储过程和函数

    概述 一提到存储过程可能就会引出另一个话题就是存储过程的优缺点,这里也不做讨论,一般别人问我我就这样回答你觉得它好你就用它.因为mysql中存储过程和函数的语法非常接近所以就放在一起,主要区别就是函数 ...

随机推荐

  1. 设计模式之复合模式(Compound Pattern)

    一.什么是复合模式? 在形式上,复合模式确实是多个模式的组合,但满足了这一条并不一定是复合模式,注意它的定义: 将多个模式结合起来形成一个“框架”,以解决一般性问题 一提到“框架”,可能最容易联想到的 ...

  2. C# 委托和事件,简单示例说明问题

    先看看示例效果 按照国际惯例,得先说说概念. 委托(C# 编程指南) 事件(C# 编程指南) 以上内容来自MSDN. 委托源码 [委托] 概念和代码都有了.剩下的就是应用了,要是只知道概念不会用,那还 ...

  3. Windows10 家庭版添加【本地组策略编辑器】

    Windows10 家庭版默认没有[本地组策略编辑器],添加方法: 新建记事本复制以下内容 @echo off pushd "%~dp0" dir /b C:\Windows\se ...

  4. VS中ReportView的坑爹问题

    ReportViewer不是.netframework提供的,而是visual studio提供的组件,它依赖如下组件: Microsoft.ReportViewer.Winforms.dll Mic ...

  5. maven項目創建紅叉

    出现问题的原因: 解决方案: 在pom.xml中配置jdk的版本

  6. 940. Distinct Subsequences II

    Given a string S, count the number of distinct, non-empty subsequences of S . Since the result may b ...

  7. python爬虫的页面数据解析和提取/xpath/bs4/jsonpath/正则(2)

    上半部分内容链接 : https://www.cnblogs.com/lowmanisbusy/p/9069330.html 四.json和jsonpath的使用 JSON(JavaScript Ob ...

  8. 队列的实现——c++

    一.介绍 队列(Queue),是一种线性存储结构.它有以下几个特点:(01) 队列中数据是按照"先进先出(FIFO, First-In-First-Out)"方式进出队列的.(02 ...

  9. easyUI取消选中的所有行

    在datagrid选择选中行进行展示后,再返回这个datagrid重新加载数据, 原来选中的数据还是保持选中状态.执行以下的方法还是如此,如图. $("#Table").datag ...

  10. [JavaScript] iframe加载完成事件

    //iframe加载完成后,对其子元素进行操作 var iframe = document.getElementById("re-img"); if (iframe.attachE ...