mysql 用存储过程和函数分别模拟序列
在其他大部分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 用存储过程和函数分别模拟序列的更多相关文章
- MySQL:存储过程和函数
存储过程和函数 一.创建存储过程和函数 1.创建存储过程 语法: CREATE PROCEDURE sp_name ([proc_parameter[,...]]) [characteristic . ...
- MYSQL的存储过程和函数简单写法
存储过程 MySQL中,创建存储过程的基本形式如下: CREATE PROCEDURE sp_name ([proc_parameter[,...]]) [characteristic ...] ro ...
- Mysql中存储过程和函数的写法
MySQL中,创建存储过程的基本形式如下: CREATE PROCEDURE sp_name ([proc_parameter[,...]]) [characteristic ...] routine ...
- MySQL之存储过程和函数
存储过程和函数: 1.创建存储过程和函数: 存储过程: delimiter $$ create procedure proc_name() BEGIN 查询语句; // 记得加分号 END $$ de ...
- mysql的存储过程,函数,事件,权限,触发器,事务,锁,视图,导入导出
1.创建过程 1.1 简单创建 -- 创建员工表 DROP TABLE IF EXISTS employee; CREATE TABLE employee( id int auto_increment ...
- mySQL查看存储过程、函数、视图、触发器
一.查看存储过程 1.show procedure status; //查看所有的 2.show create procedure proc_AllUser[proc_name]; 查看proc_Al ...
- navicat 给mysql 添加存储过程(函数)
BEGIN DECLARE i INT default 0; DECLARE num int default 0; DECLARE count1 int default 0; DECLARE coun ...
- 我的MYSQL学习心得(十) 自定义存储过程和函数
我的MYSQL学习心得(十) 自定义存储过程和函数 我的MYSQL学习心得(一) 简单语法 我的MYSQL学习心得(二) 数据类型宽度 我的MYSQL学习心得(三) 查看字段长度 我的MYSQL学习心 ...
- MySQL 存储过程和函数
概述 一提到存储过程可能就会引出另一个话题就是存储过程的优缺点,这里也不做讨论,一般别人问我我就这样回答你觉得它好你就用它.因为mysql中存储过程和函数的语法非常接近所以就放在一起,主要区别就是函数 ...
随机推荐
- Django:model.save()的时候在干什么
转:https://www.cnblogs.com/zywscq/p/5397439.html Model.save(force_insert=False, force_update=False, u ...
- ASP.NET 实现多页面合并一页显示
目前业务有一个需求: 就是把多个网页合并到一个页面显示, 在实现过程中我一般使用两种方法: 利用母版页设置导航栏, 定位到每个网页; 利用用户控件( .acsx 后缀的文件), 但是有个问题就是传参比 ...
- 程序媛计划——mysql连接表
#inner join等值连接/内连接 mysql> select * from info; +------+-------------+----------+ | name | phone | ...
- 纯文本-FileInputStream的编码与解码方式
前言:以下分析只针对纯文本 1.FileInputStream默认的编码方式就是文件的编码方式 即:源文件是什么编码方式,则利用FileInputStream默认读取的字节数组,就是什么编码方式. 例 ...
- hdu5833----高斯消元
题目大意: 给你n个整数,从中选一些数,他们的乘积为一个完全平方数 问有多少种这样的方式,已知这些数的素因素不超过2000. 思路: 一个完全平方数素因素的个数肯定是偶数个. 我们只要从n个数中选取所 ...
- tenda某路由器信息泄露查找
本文作者:i春秋作家——icqb32d3a26 1: 前期准备: (1) 路由器固件 一般获取固件的方法有以下几种 官方网站根据对应版本下载(√),点击下载 在点击更新固件时抓取对应的更新固件链接 拆 ...
- [Swift实际操作]七、常见概念-(4)范围CGRect的使用详解
本文将为你演示区域对象CGRect的使用.你可以将区域对象,看作是点对象和尺寸对象的组合 首先导入需要使用到底界面工具框架 import UIKit 然后初始化一个区域对象,它的原点位于(0,0),宽 ...
- mysql导入数据错误
ERROR 29 (HY000): File '/tmp/cun' not found (Errcode: 13) # sudo /etc/init.d/apparmor reload(重新加载) 大 ...
- 导数、多元函数、梯度、链式法则及 BP 神经网络
一元函数的导数 对于函数\(y=f(x)\),导数可记做\(f'(x_0)\).\(y'|x=x_0\)或\(\frac{dy}{dx}|x=x_0 \).定义如下: \[f'(x_0) = \lim ...
- 彻底弄懂JS的事件冒泡和事件捕获(不推荐阅读)
由于搬去敌台了,好久没来博客园,今天无意中翻到有“误认子弟”的评论,这里特意做个说明. 本文中关于事件冒泡和事件捕获的描述和例子都是OK的,错就错在后面用jquery去展示了利用事件冒泡的例子有误,其 ...