存储过程与自定义函数

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)