Mysql系列 - 第十七篇:存儲過程&自定義函數詳解

這是Mysql系列第17篇。

環境:mysql5.7.25,cmd命令中進行演示。

代碼中被[]包含的表示可選,|符號分開的表示可選其一。

需求背景介紹

線上程序有時候出現問題導致數據錯誤的時候,如果比較緊急,我們可以寫一個存儲來快速修復這塊的數據,然後再去修復程序,這種方式我們用到過不少。

存儲過程相對於java程序對於java開發來說,可能並不是太好維護以及閱讀,所以不建議在程序中去調用存儲過程做一些業務操作。

關於自定義函數這塊,若mysql內部自帶的一些函數無法滿足我們的需求的時候,我們可以自己開發一些自定義函數來使用。

所以建議大家掌握mysql中存儲過程和自定義函數這塊的內容。

本文內容

  • 詳解存儲過程的使用

  • 詳解自定義函數的使用

準備數據

/*建庫javacode2018*/
drop database if exists javacode2018;
create database javacode2018;

/*切換到javacode2018庫*/
use javacode2018;

/*建表test1*/
DROP TABLE IF EXISTS t_user;
CREATE TABLE t_user (
  id   INT NOT NULL PRIMARY KEY COMMENT '編號',
  age  SMALLINT UNSIGNED NOT NULL COMMENT '年齡',
  name VARCHAR(16) NOT NULL COMMENT '姓名'
) COMMENT '用戶表';

存儲過程

概念

一組預編譯好的sql語句集合,理解成批處理語句。

好處:

  • 提高代碼的重用性

  • 簡化操作

  • 減少編譯次數並且減少和數據庫服務器連接的次數,提高了效率。

創建存儲過程

create procedure 存儲過程名([參數模式] 參數名 參數類型)
begin
    存儲過程體
end

參數模式有3種:

in:該參數可以作爲輸入,也就是該參數需要調用方傳入值。

out:該參數可以作爲輸出,也就是說該參數可以作爲返回值。

inout:該參數既可以作爲輸入也可以作爲輸出,也就是說該參數需要在調用的時候傳入值,又可以作爲返回值。

參數模式默認爲IN。

一個存儲過程可以有多個輸入、多個輸出、多個輸入輸出參數。

調用存儲過程

call 存儲過程名稱(參數列表);

注意:調用存儲過程關鍵字是call

刪除存儲過程

drop procedure [if exists] 存儲過程名稱;

存儲過程只能一個個刪除,不能批量刪除。

if exists:表示存儲過程存在的情況下刪除。

修改存儲過程

存儲過程不能修改,若涉及到修改的,可以先刪除,然後重建。

查看存儲過程

show create procedure 存儲過程名稱;

可以查看存儲過程詳細創建語句。

示例

示例1:空參列表

創建存儲過程

/*設置結束符爲$*/
DELIMITER $
/*如果存儲過程存在則刪除*/
DROP PROCEDURE IF EXISTS proc1;
/*創建存儲過程proc1*/
CREATE PROCEDURE proc1()
  BEGIN
    INSERT INTO t_user VALUES (1,30,'路人甲Java');
    INSERT INTO t_user VALUES (2,50,'劉德華');
  END $

/*將結束符置爲;*/
DELIMITER ;

delimiter用來設置結束符,當mysql執行腳本的時候,遇到結束符的時候,會把結束符前面的所有語句作爲一個整體運行,存儲過程中的腳本有多個sql,但是需要作爲一個整體運行,所以此處用到了delimiter。

mysql默認結束符是分號。

上面存儲過程中向t_user表中插入了2條數據。

調用存儲過程:

CALL proc1();

驗證效果:

mysql> select * from t_user;
+----+-----+---------------+
| id | age | name          |
+----+-----+---------------+
|  1 |  30 | 路人甲Java    |
|  2 |  50 | 劉德華        |
+----+-----+---------------+
2 rows in set (0.00 sec)

存儲過程調用成功,test1表成功插入了2條數據。

示例2:帶in參數的存儲過程

創建存儲過程:

/*設置結束符爲$*/
DELIMITER $
/*如果存儲過程存在則刪除*/
DROP PROCEDURE IF EXISTS proc2;
/*創建存儲過程proc2*/
CREATE PROCEDURE proc2(id int,age int,in name varchar(16))
  BEGIN
    INSERT INTO t_user VALUES (id,age,name);
  END $

/*將結束符置爲;*/
DELIMITER ;

調用存儲過程:

/*創建了3個自定義變量*/
SELECT @id:=3,@age:=56,@name:='張學友';
/*調用存儲過程*/
CALL proc2(@id,@age,@name);

驗證效果:

mysql> select * from t_user;
+----+-----+---------------+
| id | age | name          |
+----+-----+---------------+
|  1 |  30 | 路人甲Java    |
|  2 |  50 | 劉德華        |
|  3 |  56 | 張學友        |
+----+-----+---------------+
3 rows in set (0.00 sec)

張學友插入成功。

示例3:帶out參數的存儲過程

創建存儲過程:

delete a from t_user a where a.id = 4;
/*如果存儲過程存在則刪除*/
DROP PROCEDURE IF EXISTS proc3;
/*設置結束符爲$*/
DELIMITER $
/*創建存儲過程proc3*/
CREATE PROCEDURE proc3(id int,age int,in name varchar(16),out user_count int,out max_id INT)
  BEGIN
    INSERT INTO t_user VALUES (id,age,name);
    /*查詢出t_user表的記錄,放入user_count中,max_id用來存儲t_user中最小的id*/
    SELECT COUNT(*),max(id) into user_count,max_id from t_user;
  END $

/*將結束符置爲;*/
DELIMITER ;

proc3中前2個參數,沒有指定參數模式,默認爲in。

調用存儲過程:

/*創建了3個自定義變量*/
SELECT @id:=4,@age:=55,@name:='郭富城';
/*調用存儲過程*/
CALL proc3(@id,@age,@name,@user_count,@max_id);

驗證效果:

mysql> select @user_count,@max_id;
+-------------+---------+
| @user_count | @max_id |
+-------------+---------+
|           4 |       4 |
+-------------+---------+
1 row in set (0.00 sec)

示例4:帶inout參數的存儲過程

創建存儲過程:

/*如果存儲過程存在則刪除*/
DROP PROCEDURE IF EXISTS proc4;
/*設置結束符爲$*/
DELIMITER $
/*創建存儲過程proc4*/
CREATE PROCEDURE proc4(INOUT a int,INOUT b int)
  BEGIN
    SET a = a*2;
    select b*2 into b;
  END $

/*將結束符置爲;*/
DELIMITER ;

調用存儲過程:

/*創建了2個自定義變量*/
set @a=10,@b:=20;
/*調用存儲過程*/
CALL proc4(@a,@b);

驗證效果:

mysql> SELECT @a,@b;
+------+------+
| @a   | @b   |
+------+------+
|   20 |   40 |
+------+------+
1 row in set (0.00 sec)

上面的兩個自定義變量@a、@b作爲入參,然後在存儲過程內部進行了修改,又作爲了返回值。

示例5:查看存儲過程

mysql> show create procedure proc4;
+-------+-------+-------+-------+-------+-------+
| Procedure | sql_mode | Create Procedure | character_set_client | collation_connection | Database Collation |
+-------+-------+-------+-------+-------+-------+
| proc4     | ONLY_FULL_GROUP_BY,STRICT_TRANS_TABLES,NO_ZERO_IN_DATE,NO_ZERO_DATE,ERROR_FOR_DIVISION_BY_ZERO,NO_AUTO_CREATE_USER,NO_ENGINE_SUBSTITUTION | CREATE DEFINER=`root`@`localhost` PROCEDURE `proc4`(INOUT a int,INOUT b int)
BEGIN
    SET a = a*2;
    select b*2 into b;
  END | utf8                 | utf8_general_ci      | utf8_general_ci    |
+-------+-------+-------+-------+-------+-------+
1 row in set (0.00 sec)

函數

概念

一組預編譯好的sql語句集合,理解成批處理語句。類似於java中的方法,但是必須有返回值。

創建函數

create function 函數名(參數名稱 參數類型)
returns 返回值類型
begin
    函數體
end

參數是可選的。

返回值是必須的。

調用函數

select 函數名(實參列表);

刪除函數

drop function [if exists] 函數名;

查看函數詳細

show create function 函數名;

示例

示例1:無參函數

創建函數:

/*刪除fun1*/
DROP FUNCTION IF EXISTS fun1;
/*設置結束符爲$*/
DELIMITER $
/*創建函數*/
CREATE FUNCTION fun1()
  returns INT
  BEGIN
    DECLARE max_id int DEFAULT 0;
    SELECT max(id) INTO max_id FROM t_user;
    return max_id;
  END $
/*設置結束符爲;*/
DELIMITER ;

調用看效果:

mysql> SELECT fun1();
+--------+
| fun1() |
+--------+
|      4 |
+--------+
1 row in set (0.00 sec)

示例2:有參函數

創建函數:

/*刪除函數*/
DROP FUNCTION IF EXISTS get_user_id;
/*設置結束符爲$*/
DELIMITER $
/*創建函數*/
CREATE FUNCTION get_user_id(v_name VARCHAR(16))
  returns INT
  BEGIN
    DECLARE r_id int;
    SELECT id INTO r_id FROM t_user WHERE name = v_name;
    return r_id;
  END $
/*設置結束符爲;*/
DELIMITER ;

運行看效果:

mysql> SELECT get_user_id(name) from t_user;
+-------------------+
| get_user_id(name) |
+-------------------+
|                 1 |
|                 2 |
|                 3 |
|                 4 |
+-------------------+
4 rows in set (0.00 sec)

存儲過程和函數的區別

存儲過程的關鍵字爲procedure,返回值可以有多個,調用時用call一般用於執行比較複雜的的過程體、更新、創建等語句

函數的關鍵字爲function返回值必須有一個,調用用select,一般用於查詢單個值並返回。

  存儲過程 函數
返回值 可以有0個或者多個 必須有一個
關鍵字 procedure function
調用方式 call select
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章