MySQL 第八篇:自定义函数、存储过程、游标

我把MySQL的内容整理成9篇博客,学完这9篇博客虽不能说能成为大神,但是应付一般中小企业的开发已经足够了,有疑问或建议的欢迎留言讨论。

自定义函数

一、函数的概念与定义

1、理解函数

函数可以看作是一个加工作坊,这个加工作坊接收调用者传递过来的原料(传递的参数),然后将这些原料加工处理成产品(函数的返回值)再把产品返回给调用者。

2、创建语法(建议自定义函数以fun_前缀)

create function 函数名(参数1,参数2....) returns 返回值的数据类型
[函数类型]
begin
函数体
return 语句;
end;
-- 将日期转换成2017年10月29日
-- DELIMITER $$的作用是声明sql语句以$$结束,也可以把$$换成其他符号,在这里可以省略,如果是命令行运行就一定要加上。
DELIMITER $$
create function fun_dateToStr(mdate datetime)returns varchar(50)
BEGIN
    -- 声明一个变量 使用declare 变量名 数据类型 默认值
    declare str varchar(50) default '';
    -- 使用set 对变量进行赋值
    set str = DATE_FORMAT(mdate,'%Y年%m月%d日');
    return str;
end $$
DELIMITER ;
-- 使用函数
select *,fun_dateToStr(NOW()) from students

二、自定义函数

1、DELIMITER

定义一个结束标识符,因为MySQL默认是以分号作为SQL语句的结束符的,而函数体内部要用到分号,所以会跟默认的SQL结束符发生冲突,所以需要先定义一个其他的符号作为SQL的结束符;

2、分支结构

IF 条件 THEN 语句 ELSEIF 条件 THEN 语句 ELSE 语句 END IF;
DELIMITER $$
CREATE FUNCTION FUN_getAgeStr(age int) RETURNS varchar(20)
BEGIN
	declare results varchar(20);
	IF age<16 then
		set results = '小屁孩';
	ELSEIF age <22 THEN
		set results = '小鲜肉';
	ELSEIF age <30 THEN
		set results = '小青年';
	ELSE
		SET results = '大爷';
	END IF;
	RETURN results;
end $$
DELIMITER ;

select *,FUN_getAgeStr(age) from students;

3、循环结构

循环名:loop end loop 循环名 WHILE,REPEAT
DELIMITER $$
create function fun_loop(nums int) returns varchar(255)
BEGIN
	DECLARE i int default 0;
	DECLARE result varchar(255) default '';
	-- 循环的开始,冒号前面是一个循环名
	colorLoop:LOOP
	SET i = i+1;
	SET result = concat(result,'Color  ');
	-- 跳出循环
	IF i >= nums THEN
		LEAVE colorLoop;
	END IF;
	-- 结束循环
	END LOOP colorLoop;
	return result;
end $$
DELIMITER ;
select fun_loop(4)

4、删除自定义函数

DROP FUNCTION IF EXISTS 自定义函数名
drop function if exists FUN_getAgeStr;

5、查看自定义函数定义

SHOW CREATE FUNTION 自定义函数名;

6、查看自定义函数状态

SHOW FUNCTION STATUS [LIKE]

存储过程

一、创建语法(建议自定义函数以pro_前缀)

create procedure 存储过程名(参数1,参数2…)
[存储过程选项]
begin
存储过程语句块
end;
说明:

  • IN 输入参数:表示该参数的值必须在调用存储过程时指定,在存储过程中修改该参数的值不能被返回,为默认值。
  • OUT 输出参数:该值可在存储过程内部被改变,并可返回。
  • INOUT 输入输出参数:调用时指定,并且可被改变和返回。
create procedure pro_dealStudents(in s_id varchar(50))
BEGIN
    set @s_id = s_id;
    select * from students s where s.s_id=@s_id;
end;
-- 使用存储过程
CALL pro_dealStudents('J1604028');

二、存储过程与函数的区别

存储过程是用户定义的一系列sql语句的集合,涉及特定表或其它对象的任务,用户可以调用存储过程,而函数通常是数据库已定义的方法,它接收参数并返回某种类型的值并且不涉及特定用户表。
存储过程和函数存在以下几个区别:
1)一般来说,存储过程实现的功能要复杂一点,而函数的实现的功能针对性比较强。存储过程,功能强大,可以执行包括修改表等一系列数据库操作;用户定义函数不能用于执行一组修改全局数据库状态的操作。
2)对于存储过程来说可以返回参数,如记录集,而函数只能返回值或者表对象。函数只能返回一个变量;而存储过程可以返回多个。存储过程的参数可以有IN,OUT,INOUT三种类型,而函数只能有IN类~~存储过程声明时不需要返回类型,而函数声明时需要描述返回类型,且函数体中必须包含一个有效的RETURN语句。
3)存储过程,可以使用非确定函数,不允许在用户定义函数主体中内置非确定函数。
4)(mysql除外)存储过程一般是作为一个独立的部分来执行( EXECUTE 语句执行),而函数可以作为查询语句的一个部分来调用(SELECT调用),由于函数可以返回一个表对象,因此它可以在查询语句中位于FROM关键字的后面。 SQL语句中不可用存储过程,而可以使用函数。

游标

游标本质上是一种能从select结果集中每次提取一条记录的机制,因此游标与select语句息息相关。

一、使用游标的步骤:

1.声明游标 declare 游标名 cursor for select语句。
2.打开游标 open 游标名。
3.从游标中提取数据 fetch 游标名 into 标量(需配合循环使用)。
4.关闭游标 close 游标名称。
注意:
1.变量名的个数必须与声明游标时使用的select语句结果集中的字段个数保持一致
2.fetch 在执行过程中如果无法提取数据会产生 “ERROR 1329(0200):Nodata to FETCH”,这样我们可以自定义1329错误来结束遍历。

create PROCEDURE pro_students()
BEGIN
    declare s_age int ;
    declare s_no varchar(50);
    declare DONE INT DEFAULT 0;
    declare cur CURSOR for select age,s_id from students;
    DECLARE CONTINUE HANDLER FOR SQLSTATE '02000' SET DONE=1;
    open cur;
    fetch cur into s_age,s_no;
    repeat
            SET s_age = s_age+1;
        select s_age;
        update students s set s.age = s_age where s.s_id=s_no;
    fetch cur into s_age,s_no;
    UNTIL DONE END REPEAT;
close cur;
end;
CALL pro_students();
时间: 2024-10-25 18:41:18

MySQL 第八篇:自定义函数、存储过程、游标的相关文章

SQL server使用自定义函数以及游标_MsSql

编号 标准宗地编码(landCode) 所在区段编码(sectCode) 1 131001BG001 G001 2 131001BG002 G001 3 131001BG003 G001 4 131001BG004 G002 5 131001BG005 G003 现在需要将表中的数据转换为如下表所示结果: 编号 区段编码 包含的标准宗地 1 G001 131001BG001,131001BG002,131001BG003 2 G002 131001BG004 3 G003 131001BG005

SQL server使用自定义函数以及游标

编号 标准宗地编码(landCode) 所在区段编码(sectCode) 1 131001BG001 G001 2 131001BG002 G001 3 131001BG003 G001 4 131001BG004 G002 5 131001BG005 G003 现在需要将表中的数据转换为如下表所示结果: 编号 区段编码 包含的标准宗地 1 G001 131001BG001,131001BG002,131001BG003 2 G002 131001BG004 3 G003 131001BG005

Delphi FireDAC 下的 Sqlite(八) 自定义函数

Sqlite 本身没有这个功能, FireDAC 通过 TFDSQLiteFunction 增加了该功能; 尽管通过某些 SQL 语句或通过视图也可以达到类似效果, 但函数会更灵活些. 本例先建了一个成绩表, 然后通过两个 TFDSQLiteFunction 实现了 "总分" 与 "平均分" 的计算. 你可以复制下面文本框中的内容, 然后直接往窗体上贴, 以快速完成窗体设计: object DBGrid1: TDBGrid Left = 8 Top = 88 Wid

mysql 序列号生成器 (自定义函数)

1.创建生成多个表的序列号的数据维护表 Java代码   CREATE TABLE `seq` (     `name` varchar(20) NOT NULL,     `val` int(10) unsigned NOT NULL,     PRIMARY KEY (`name`)   ) ENGINE=MyISAM DEFAULT CHARSET=utf8;   2.插入几条初始化数据 Java代码   INSERT INTO seq VALUES('one',100);   INSER

SQL Server 中 自定义函数 和 游标 应用的经典案例

server|函数|游标     这是网友的问题,我当时立马给出了自己的解决方案,但是没有想到中间有点小问题,发现后经过自己仔细调试,完全得到正确结果后,那个网友已经结帖了.我的代码遂成为鸡肋,食之无味,弃之可惜.但是我觉得我的代码确实还是挺经典的,所以整理了一下,供各位网友欣赏.问题:  假设环境如下:     表1:      ID, NAME,      QQ,     PHONE, 表中数据:      1       秦云        10102800 13500000       

深入mysql创建自定义函数与存储过程的详解_Mysql

一 创建自定义函数在使用mysql的过程中,mysql自带的函数可能不能完成我们的业务需求,这时就需要自定义函数,例如笔者在开发过程中遇到下面这个问题:mysql表结构如下 复制代码 代码如下: DROP TABLE IF EXISTS `test`;CREATE TABLE `test` (  `id` int(10) unsigned NOT NULL AUTO_INCREMENT,  `pic` varchar(50) NOT NULL,  `hashcode` varchar(16) N

postgresql-求大神把PostgreSQL 一个存储过程转变为自定义函数

问题描述 求大神把PostgreSQL 一个存储过程转变为自定义函数 存储过程如下: create procedure calcGPAbySno ( @vsno char(10), @avg_gpa decimal(10,2) out ) as declare @gpa decimal(10,1); declare @vcredit int; declare @vgrade int; declare @sum_credit int; declare @sum_gpa decimal(10,1);

Mysql学习笔记(十)存储过程与函数 + 知识点补充(having与where的区别)

原文:Mysql学习笔记(十)存储过程与函数 + 知识点补充(having与where的区别) 学习内容:存储程序与函数...这一章学的我是云里雾里的... 1.存储过程...   Mysql存储过程是从mysql 5.0开始增加的一个新功能.存储过程的优点其实有很多,不过我觉得存储过程最重要的优点就是实现了SQL代码的封装,那么我们为什么需要封装SQL语句呢?原因就是当我们在面对一个庞大的数据库的时候,当我们使用外部程序去访问数据库的时候...我们总不能在外部程序中内嵌很多的SQL语句吧...

Oracle存储过程和自定义函数详解_oracle

概述 PL/SQL中的过程和函数(通常称为子程序)是PL/SQL块的一种特殊的类型,这种类型的子程序可以以编译的形式存放在数据库中,并为后续的程序块调用. 相同点: 完成特定功能的程序 不同点:是否用return语句返回值. 举个例子: create or replace procedure PrintStudents(p_staffName in xgj_test.username%type) as cursor c_testData is select t.sal, t.comm from