MariaDB MariaDB、MySQL存储过程、游标基础应用举例说明
MariaDB、MySQL存储过程、游标基础应用举例说明
by:授客 QQ:1033553122
测试环境:
MariaDB-10.0.19-centos7-x86_64
实践操作:
# 创建测试数据库
DROP DATABASE IF EXISTS testdb1;
CREATE DATABASE testdb1;
# 创建测试表-余额表
DROP TABLE IF EXISTS account;
CREATE TABLE IF NOT EXISTS account
(user_id BIGINT PRIMARY KEY, # 用户id
account BIGINT # 账户余额
);
# 创建测试表-创建利润分成表
CREATE TABLE IF NOT EXISTS profit_sharing
(order_id BIGINT PRIMARY KEY, # 订单ID
user_id BIGINT, # 用户ID
total_price DECIMAL(9,4), # 交易价格
share_price DECIMAL(9,4), # 消费返利
FOREIGN KEY(user_id) REFERENCES account(user_id)
);
# 插入测试数据
INSERT INTO account VALUES(1001, 10000);
INSERT INTO account VALUES(1002, 9000);
INSERT INTO profit_sharing VALUES(100001, 1001, 99, 10);
INSERT INTO profit_sharing VALUES(100002, 1002, 90, 5);
# 场景1:存储过程中的每个查询语句都只返回一条记录
# 创建存储过程
DROP PROCEDURE IF EXISTS proc_varify_profit_sharing; # 如果存在proc_varify_profit_sharing则先删除
DELIMITER // # 使用DELIMITER命令来把语句定界符从 ; 变为 // # 注意,定界符可以是其它符号,比如 $ ,但是不能为 \ ,因为它是转义字符
#说明,通常情况下,mysql遇到分号 ; 则判断语句结束,并执行该语句,这里DELIMITER // 告诉mysql,遇到 // 则表示语句结束,执行语句
CREATE PROCEDURE proc_varify_profit_sharing(
user_id BIGINT,
account_init BIGINT,
order_id BIGINT
)
BEGIN
# 声明局部变量
DECLARE account_final BIGINT; # 用于存储最终余额
DECLARE total_price DECIMAL(9, 4); # 用于存储交易总金额
DECLARE share_price DECIMAL(9, 4) DEFAULT 0.00; # 用于存储消费返利金额
# 说明:BEGIN END;之间定义的变量为局部变量,基础形式:DECLARE 变量名 数据类型 默认值
# 定义游标
DECLARE cur_get_account_final CURSOR FOR SELECT t1.account FROM account t1 WHERE t1.user_id=user_id;
DECLARE cur_varify_account CURSOR FOR SELECT t2.share_price, t2.total_price FROM profit_sharing t2
WHERE t2.user_id=user_id AND t2.order_id = order_id;
# 基础语法说明:DECLARE cursor_name CURSOR FOR select_statement;
# 打开游标
OPEN cur_get_account_final;
OPEN cur_varify_account;
#基础语法说明:OPEN cursor_name;
# 获取数据
FETCH cur_get_account_final INTO account_final;
FETCH cur_varify_account INTO share_price, total_price;
#基础语法说明:FETCH cursor_name INTO variable1[, variable2,...]; #读取一行记录(如果有的话),
#并将记录中各项的值赋值给变量variable1,variable2,...
#注意,变量个数要和游标指定查询生成记录的记录项的数目要相同,通俗的说,查询生成记录,包含多少列,就指定多少个变量,一一对应
# 验证金额增减的准确性
SET @tmp = account_final + total_price - share_price; #使用用户变量@tmp,存放中间计算结果
SELECT IF(@tmp=account_init, 'right', 'wrong') AS 'result';
#说明:用户变量(用户变量也归属局部变量)的定义:@变量名,,
#注意:
#1.mysql中用户变量可不用事前声明,在用的时候直接用“@变量名”使用就可以了,
#2.用户变量跟mysql客户端是绑定的,设置的变量,只对当前用户使用的客户端生效
#说明:用SET给变量赋值: SET 变量=value; 或者 SET 变量:=value; 如上语句所示
小知识:可以用SELECT给变量赋值,类似的有,SELECT 变量:=value; 或 SELECT 变量:=colunm_name FROM table_name WHERE …;
注意:使用SELECT时必须用 :=
# 关闭游标
CLOSE cur_get_account_final;
CLOSE cur_varify_account;
#基础语法说明:CLOSE cursor_name;
END;
//
DELIMITER ; # 还原定界符为 ;
# 调用存储过程
CALL proc_varify_profit_sharing(1001, 10089, 100001);
#调用形式说明:CALL procedure_name()(无参数的话)、CALL procedure_name(arg1[, arg2, ...])(带参数)
# "场景1"不用游标的等价实现
# 创建存储过程
DROP PROCEDURE IF EXISTS proc_varify_profit_sharing;
DELIMITER //
CREATE PROCEDURE proc_varify_profit_sharing(
userID BIGINT, # 注意,当存储过程参数如果用于存储过程中,表查询语句的WHERE子句,作为筛选条件值时,如果未指定要查询表的表别名,则不能和表的列名相同,参见如下
account_init BIGINT,
orderID BIGINT
)
BEGIN
# 定义局部变量
DECLARE account_final BIGINT; # 用于存储最终余额
DECLARE total_price1 DECIMAL(9, 4); # 用于存储交易总金额
DECLARE share_price1 DECIMAL(9, 4); # 用于存储消费返利金额
#获取数据
SELECT account INTO account_final FROM account WHERE user_id = userID LIMIT 0,1; # LIMIT 0, 1 确保只返回一条记录
# 把share_price, total_price分别的查询结果分别赋值给局部变量share_price, total_price
SELECT share_price, total_price INTO share_price1, total_price1 FROM profit_sharing
WHERE user_id=userID AND order_id = orderID LIMIT 0,1;
#注意:
#1、未使用游标的情况下,查询语句使用表别名需要添加 AS,如 SELECT t1.account INTO account_final FROM account t1; 报错,提示t1为 Undeclared variable ,SELECT t1.account INTO account_final FROM account AS t1则不会报错
#2、SELECT column1, ... INTO variable1, ...; 如果column1和variable1名称不可以相同,否则取不到值,variable1最终的值为NULL(MariaDB下测试
# 验证金额增减的准确性
SET @tmp = account_final + total_price1 - share_price1; # 注意:NULL + value1 + ... = NULL
SELECT IF(@tmp=account_init, 'right', 'wrong') AS 'result';
END;
//
DELIMITER ;
CALL proc_varify_profit_sharing(1001, 10089, 100001);
# 场景2:返回一个结果集,多条记录(循环读取
# 创建存储过程
DROP PROCEDURE IF EXISTS proc_test;
DELIMITER //
CREATE PROCEDURE proc_test()
BEGIN
# 声明局部变量
DECLARE order_id BIGINT; # 用于存储订单号
DECLARE total_price DECIMAL(9, 4); # 用于存储交易总金额
DECLARE share_price DECIMAL(9, 4) DEFAULT 0.00; # 用于存储消费返利金额
#DECLARE no_more_record INT DEFAULT 0; # 是否还有记录标识
DECLARE no_more_record INT; # 是否还有记录标识
# 定义游标
DECLARE cur_get_order_info CURSOR FOR SELECT share_price, total_price FROM profit_sharing;
DECLARE CONTINUE HANDLER FOR NOT FOUND SET no_more_record = 1; #当游标或者SQL选择语句没有返回值时,设置 no_more_record=1后继续执行
# 打开游标
OPEN cur_get_order_info;
# 读取数据 ======行1
FETCH cur_get_order_info INTO total_price, share_price;
SET no_more_record = 0;
WHILE no_more_record <> 1 DO #
如果存在下一跳记录,则继续读取
FETCH cur_get_order_info INTO total_price, share_price;
END WHILE;
# ===============行2
#注意:如果把行1到行2之间的语句改成如下
#WHILE no_more_record <> 1 DO #
如果存在下一跳记录,则继续读取
#
FETCH cur_get_order_info INTO total_price, share_price;
#END WHILE;
#那么,整个while循环体将执行3次(但是表里仅2条记录,为何这样呢?个人想法是这样的:
#第一次循环,no_more_record=0,执行FETCH语句,且游标指针停留在一条记录
#第二次循环,no_more_record=0,执行FETCH语句,且游标指针停留在二条记录
#第三次循环,no_more_record=0,执行FETCH语句,且游标指针移动到下一条,结果无记录,no_more_record=1,结束下次循环
#所以,while循环之前要先执行一次
#
关闭游标
CLOSE cur_get_order_info;
END;
//
DELIMITER ;
CALL proc_test();
MariaDB MariaDB、MySQL存储过程、游标基础应用举例说明的更多相关文章
- MySQL存储过程 游标
MySQL存储过程 游标 如何在存储过程中使用MySQL游标来遍历SELECT语句返回的结果集 MySQL游标简介 要处理存储过程中的结果集,请使用游标.游标允许您迭代查询返回的一组行,并相应地处理 ...
- mysql存储过程游标嵌套循环
自己写的一个mysql存储过程如下: BEGIN DECLARE _did bigint(20);DECLARE _count int;DECLARE s1 int;DECLARE cur_1 CUR ...
- ava基础MySQL存储过程 Java基础 JDBC连接MySQL数据库
1.MySQL存储过程 1.1.什么是存储过程 带有逻辑的sql语句:带有流程控制语句(if while)等等 的sql语句 1.2.存储过程的特点 1)执行效率非常快,存储过程是数据库的服 ...
- MySQL 存储过程/游标/事务
将会用到的几个表 mysql> DESC products; +------------+--------------+------+-----+---------+-------------- ...
- php调用mysql存储过程游标
<?php $dbtype = 'mysql'; $host = 'localhost'; $dbname = 'test'; $dsn = "$dbtype:host=$host;d ...
- mysql存储过程游标加计划任务事件调度器
存储过程加事件调度器 -- 存储过程 (多个)游标的使用 临时表的使用(让执行时间从一个小时降低到5分钟)DELIMITER $$ DROP PROCEDURE IF EXISTS `eval_cal ...
- MySQL存储过程入门基础
创建存储过程无参语法: delimiter // create procedure 函数名() begin 业务逻辑 end // call 函数名() 通过函数名调用存储过程 创建存储过程有参与法: ...
- mysql 存储过程 游标嵌套
基本表temp 包括 name, type, sendCity, getCity 分别对应物流送货司机名, 倒车的第几段, 发货城市, 收货城市 表结构 -- -------------------- ...
- MySQL 存储过程游标
一.创建游标 游标用declare语句创建.如下面的例子所示: create procedure test2() begin declare cursorTest cursor for select ...
随机推荐
- odoo开发笔记 -- context上下文
字段级别 视图级别 窗口动作级别
- 线程中的定时器Timer类
Timer 定时器 几分钟之后执行一个任务. 创建了一个定时器相当于开启了一条线程,TimerTask相当于一个线程的任务.内部使用wait/notify机制来实现的. 用法非常的简单 就足以里面的 ...
- Nginx 配置 Location 规则优先级问题
nginx 的 location与配置中 location 顺序没有关系,与 location 表达式的类型有关.相同类型的表达式,字符串长的会优先匹配. 以下是按优先级排列说明: 等号类型(=)的优 ...
- 源码分析篇 - Android绘制流程(三)requestLayout()与invalidate()流程及Choroegrapher类分析
本文主要探讨能够触发performTraversals()执行的invalidate().postInvalidate()和requestLayout()方法的流程.在调用这三个方法到最后执行到per ...
- Method 'initializationerror' not found.Opening the test classs JUnit4单元测试报错问题解决办法(图文详解)
不多说,直接上干货! 问题现象 今天使用JUnit 4进行单元测试时,测试程序一直运行不起来,报method initializationerror not found错误,如下: 问题分析 网上说版 ...
- WPF 中textBox实现只输入数字
刚学到 通过本方法可以使文本框只能输入或复制入数字 对于数量类输入文本框比较有用 金额类只需小改动也可实现 以TextBox txtCount为例 添加TextChanged事件 代码如下 priv ...
- java学习-Comparable<Integer>接口方法的实现
基本数据类型的包装类Integer, Float, Double,Long,Byte等都实现的Comparable接口,用于列表List或数组arrays的排序 Comparable<Integ ...
- ElasticSearch 基础<转载>
使用curl命令操作elasticsearch 大岩不灿 发表于 2015年4月25日 浏览 13,463 次 第一:_cat系列_cat系列提供了一系列查询elasticsearch集群状态的接口. ...
- vim 常用命令(一)特殊删除
dd 可以删除光标当前行: ndd n代表行数,3dd,删除从当前光标往下3行: di 删除在指定符号内的内容,如 空号,引号内: dt 删除当前光标到指定符号的内容
- WPF中路由事件的传播
路由事件(RoutedEvent)是WPF中新增的事件,使用起来与传统的事件差别不大, 但传播方式是完全不同的. 路由事件的传播方式 通过RoutingStrategy来定义传播的方式 public ...