存储过程与自定义函数
1 概念与区别
含义:一组预先编译好的SQL语句的集合,理解成批处理语句
区别:
- 存储过程:可以有0个返回,也可以有多个返回,适合做批量插入、批量更新
- 函数:有且仅有1 个返回,适合做处理数据后返回一个结果
2 存储过程
2.1 创建与调用
语法:
CREATE PROCEDURE 存储过程名(参数列表)
BEGIN
存储过程体(一组合法的SQL语句;
END 结束标记
CALL 存储过程名(实参列表);
注意:
1)参数列表包含三部分:参数模式 参数名 参数类型,如 in stuname varchar(20)
2)参数模式
- in:该参数可以作为输入,也就是该参数需要调用方传入值
- out:该参数可以作为输出,也就是该参数可以作为返回值
- inout:该参数既可以作为输入又可以作为输出,也就是该参数既需要传入值,又可以返回值
3)如果存储过程体仅仅只有一句话,begin end可以省略
4)存储过程体中的每条sql语句的结尾要求必须加分号
5)存储过程的结尾可以使用 delimiter 重新设置,语法:delimiter 结束标记,如:delimiter $
案例:
#1.空参列表
#案例1:插入到admin表中五条记录
SELECT * FROM admin;
DELIMITER $
CREATE PROCEDURE myp1()
BEGIN
INSERT INTO admin(username,`password`)
VALUES('john1','0000'),('lily','0000'),('rose','0000'),('jack','0000'),('tom','0000');
END $
#调用
CALL myp1()
# 2.创建带in模式参数的存储过程
#案例2 :创建存储过程实现,用户是否登录成功
DELIMITER $
CREATE PROCEDURE myp3(IN username VARCHAR(20),IN PASSWORD VARCHAR(20))
BEGIN
DECLARE result INT DEFAULT 0;#声明并初始化
SELECT COUNT(*) INTO result#赋值
FROM admin
WHERE admin.username = username
AND admin.password = PASSWORD;
SELECT IF(result>0,'成功','失败') 是否登录成功;#使用
END $
#调用
CALL myp3('Jake','8888')
#3.创建out模式参数的存储过程
#案例3:根据输入的女神名,返回对应的男神名和魅力值
DELIMITER $
CREATE PROCEDURE myp5(IN beautyName VARCHAR(20),OUT boyName VARCHAR(20),OUT usercp INT)
BEGIN
SELECT boys.boyname ,boys.usercp INTO boyname,usercp
FROM boys
RIGHT JOIN beauty b
ON b.boyfriend_id = boys.id
WHERE b.name=beautyName ;
END $
#调用
CALL myp5('Lilia',@name,@cp)
SELECT @name,@cp
#4.创建带inout模式参数的存储过程
#案例4:传入a和b两个值,最终a和b都翻倍并返回
DELIMITER $
CREATE PROCEDURE myp6(INOUT a INT ,INOUT b INT)
BEGIN
SET a=a*2;
SET b=b*2;
END $
#调用
SET @m=10;
SET @n=20;
CALL myp6(@m,@n);
SELECT @m,@n;
2.2 删除与查看信息
(1)删除存储过程
语法:drop procedure 存储过程名
只允许单个删除,不支持多个删除
DROP PROCEDURE myp4;
(2)查看存储过程的信息
SHOW CREATE PROCEDURE myp2;
2.3 案例
# 案例1:创建存储过程实现传入用户名和密码,插入到admin表中
DELIMITER $
CREATE PROCEDURE myp7(IN username VARCHAR(20) , IN mypassword VARCHAR(20))
BEGIN
INSERT INTO admin(admin.username,admin.password)
VALUES(username,mypassword);
END $
CALL myp7('Lilia','980426')
SELECT * FROM admin
# 案例2:创建存储过程或函数实现传入一个日期,格式化成××年××月××日并返回
DELIMITER $
CREATE PROCEDURE myp10(IN mydata DATETIME,OUT strdate VARCHAR(50))
BEGIN
SELECT DATE_FORMAT(mydata,'%y年%m月%d日') INTO strdate;
END $
CALL myp10(NOW(),@str)
SELECT @str
3 自定义函数
3.1 创建与调用
语法:
CREATE FUNCTION 函数名(参数列表) RETURNS 返回类型
BEGIN
函数体
END
SELECT 函数名(参数列表)
注意:
1)参数列表 包含两部分:参数名 参数类型
2)函数体:肯定会有return语句,如果没有会报错;如果return语句没有放在函数体的最后也不报错,但不建议;return 值;
3)函数体中仅有一句话,则可以省略begin end
4)使用 delimiter语句设置结束标记
3.2 删除与查看
# 删除函数
DROP FUNCTION myf3;
# 查看函数
SHOW CREATE FUNCTION myf3;
3.3 案例
#1.无参有返回
# 如果我们开启了 bin-log, 我们就必须为我们的function指定一个参数。
# 则需要下面语句进行解决
SET GLOBAL log_bin_trust_function_creators=TRUE;
#案例1:返回公司的员工个数
DELIMITER $
CREATE FUNCTION myf1() RETURNS INT
BEGIN
DECLARE c INT DEFAULT 0;#定义局部变量
SELECT COUNT(*) INTO c#赋值
FROM employees;
RETURN c;
END $
SELECT myf1() 员工个数
#2.有参有返回
#案例2:根据部门名,返回该部门的平均工资
DELIMITER $
CREATE FUNCTION myf3(deptName VARCHAR(20)) RETURNS DOUBLE
BEGIN
DECLARE sal DOUBLE ;
SELECT AVG(salary) INTO sal
FROM employees e
JOIN departments d ON e.department_id = d.department_id
WHERE d.department_name=deptName;
RETURN sal;
END $
SELECT myf3('IT') IT部门平均工资
#案例3:创建函数,实现传入两个float,返回二者之和
DELIMITER $
CREATE FUNCTION test_fun1(num1 FLOAT,num2 FLOAT) RETURNS FLOAT
BEGIN
DECLARE SUM FLOAT DEFAULT 0;
SET SUM=num1+num2;
RETURN SUM;
END $
SELECT test_fun1(1,2)