目录

引言

创建存储过程和存储函数

创建存储过程

创建存储函数 

变量的应用

局部变量

全局变量

为变量赋值 

光标的运用

声明光标(declare cursor)

打开光标(open cursor)

使用光标(fetch cursor)

关闭光标(close cursor)

流程控制语句

if语句

 case语句

while循环语句


引言

        存储过程和函数:在数据库中定义一些SQL语句的集合,然后直接调用这些存储过程和函数来执行已经定义好的SQL语句。

        存储过程和函数的作用:可以避免开发人员重复的编写相同的SQL语句;由于存储过程和函数是在MySQL服务器中存储和执行的,因此可以减少客户端和服务端的数据传输。

        本文将介绍存储过程和函数的含义、作用,以及创建、使用、查看、修改和删除存储过程和函数的方法。

创建存储过程和存储函数

        通常在数据库系统中,为了保证数据的完整性一致性,同时为了提高应用性能,大多数据库常采用存储过程存储函数技术。

        存储过程和存储函数是一组SQL语句的组合,被当作整体存入MySQL数据库服务器中。

        用户定义的存储函数不能用于修改全局库状态,但该函数可以从查询中被唤醒调用,也可以像存储过程一样通过语句执行。

创建存储过程

在MySQL中,创建存储过程,语法格式如下:

create procedure sp_name ([ in | out | inout ] param_name type,......)

[characteristic ...... ]

begin

......

end;

参数说明:

        sp_name:存储过程名称

        in:输出参数

        out:输出参数        

        inout:既可以输入也可以输出

        param_name:存储过程参数名称

        type:指定存储过程的参数类型,该类型可以是MySQL数据库的任意数据类型

        characteristic:指定存储过程特性

        begin:开始

        end:结束

※注意

        由于存储过程内部语句要以分号结束,所以在定义存储过程前,应将语句结束标志“;”更改为其他字符,并且应降低该字符在存储过程中出现的概率,更改结束标志可以用关键字“delimiter”定义。

mysql >delimiter//
#创建一个名称未count_of_student的存储过程。首先,创建一个名为students的MySQL数据库,然后创建一个名未studentinfo的数据表
DELIMITER $$ #将语句结束标志设置为 ‘$$’

USE `students`$$ 

DROP PROCEDURE IF EXISTS `count_of_student`$$

CREATE DEFINER=`root`@`localhost` PROCEDURE `count_of_student`(OUT count_num INT)
    READS SQL DATA
BEGIN
	SELECT COUNT(*) INTO count_num FROM studentinfo;

	END$$

DELIMITER ;#将语句结束标志设置为 ‘;’

创建存储函数 

        创建存储函数与创建存储过程大体相同。其创建存储函数的基本形式如下:

create function sp_name (func_parameter[,......])

returns type

[characteristic......] routine_body

参数

说明

参数

说明

sp_name

存储函数的名称

characteristic

指定存储过程的特性

fun_parameter

存储函数的参数列表

routine_body

SQL代码的内容

returns type

指定返回值的类型

func_parameter由多个参数组成,其中每个参数均由参数名称和参数类型组成,结构如下:

        parameter_name type

        paramter_name:是存储函数的函数名称,type参数用于指定存储函数的参数类型

DELIMITER $$

USE `students`$$

DROP FUNCTION IF EXISTS `name_of_student`$$

CREATE DEFINER=`root`@`localhost` FUNCTION `name_of_student`(std_id INT) RETURNS VARCHAR(50) CHARSET utf8
BEGIN
	RETURN (SELECT  NAME FROM studentinfo WHERE sid=std_id);
    END$$

DELIMITER ;

变量的应用

        在MySQL存储过程中的参数主要有局部参数和会话参数两种,这两种参数又可以被称为局部变量和会话变量。局部变量只能定义该局部变量的begin......end范围内有效,会话变量在整个存储过程范围内均有效。

局部变量

        语法格式如下:

declare var_name[,......] type[default value]

        局部变量以关键字declare声明,后跟变量名和变量类型:

declare a int

        当然在声明局部变量时可以用关键字default为变量指定默认值:
declare a int default 10

#分别在内层和外层begin……end块中都定义同名的变量x,按照语句从上到下执行的顺序,
如果变量x在整个程序中都有效,则最终结果应该都为inner,
但真正的输出结果却不同,这说明在内部begin……end块中定义的变量只在该块内有效。
DELIMITER $$

USE `students`$$

DROP PROCEDURE IF EXISTS `p1`$$

CREATE DEFINER=`root`@`localhost` PROCEDURE `p1`()
BEGIN
	DECLARE X CHAR(10) DEFAULT 'outer';
	BEGIN
	DECLARE X CHAR(10) DEFAULT 'inner';
	SELECT X;
	END;
	SELECT X;
	

	END$$

DELIMITER ;
call p1();

mysql存储过程打不开了 mysql存储过程cursor_存储过程

mysql存储过程打不开了 mysql存储过程cursor_mysql_02

全局变量

        会话变量不必声明即可使用,会话变量在整个过程中有效,会话变量名以字符“@”作为起始字符。

#在该示例中,分别在内部和外部begin.....end块中都定义了同名的会话变量@t,
并且最终输出结果相同,说明会话变量的作用范围为整个程序

DELIMITER $$

USE `students`$$

DROP PROCEDURE IF EXISTS `p2`$$

CREATE DEFINER=`root`@`localhost` PROCEDURE `p2`()
BEGIN
	SET @t=1;
	BEGIN 
	SET @t=2;
	SELECT @t;
	END;
	SELECT @t;

	END$$

DELIMITER ;
call p2();

mysql存储过程打不开了 mysql存储过程cursor_存储过程_03

mysql存储过程打不开了 mysql存储过程cursor_存储过程_04

为变量赋值 

        使用set关键字为变量赋值。set语句的语法格式如下:

set var_name =expr[,var_name=expr].........

set关键字:用来为变量赋值

var_name:变量的名称

expr:赋值表达式

多个变量赋值时用“,”隔开

例如:

set mr_soft=10;

另外,MySQL中还可以应用另一种方式为变量赋值。其语法结构如下:

select col_name[,......] into var_name [,......] from table_name where condition;

col_name:表示查询的字段名称

var_name:变量的名称

table_name:指定数据表名称

condition:指定查询条件

例如:

#从studentinfo表中查询 name为'LeonSK'的记录 ,将该记录下的tel字段内容
赋值给变量customer_tel

select tel into customer_tel from studentinfo where name='leonSK';

※注意

        上述赋值语句必须存在于创建的存储过程中,且需将赋值语句放在begin......end之间。若脱离此范围,该变量将不能被使用或被赋值。

光标的运用

        通过数据库查询,其结果可能是多条记录。在存储过程和函数中使用光标可以实现逐条读取结果集中的记录。光标的使用包括:

        声明光标(declare cursor)

        打开光标(open cursor)

        使用光标(fetch cursor)

        关闭光标(close cursor)

声明光标(declare cursor)

在MySQL中,声明光标仍然使用declare关键字,语法如下:

declare cursor_name cursor for select_statement;

cursor_name:光标的名称,其使用与表名同样的规则

select_statement:是一个select语句,返回一行或多行数据。

例如:

#定义光标info_of_student

declare info_of_student cursor for select
sid,name,age,sex,age
from studentinfo
where sid=1;

※注意

        这个语句也可以在存储过程中定义多个光标,但是必须保证每个光标名称的唯一性,即每个光标需要有自己唯一的名称。

        这里的select子句不能包含into 子句,并且光标只能在存储过程或存储函数中使用,不能单独执行。

打开光标(open cursor)

        在声明光标之后,要从光标中提取数据,必须要先打开光标,语法格式如下

open cursor_name;

在程序中一个光标可以打开多次,因为由于用户打开光标后,其他用户或程序正在更新数据表,导致用户在每次打开光标后,显示的结果都不相同。

#打开贯标info_of_student

open info_of_student

使用光标(fetch cursor)

        在光标顺利打开后,可以使用fetch....into 语句来读取数据。语法格式如下:

fetch cursor_name into var_name [,var_name]..........

cursor_name:代表已经打开光标的名称

var_name:表示将光标中select语句查询出来的参数存入该参数中。var_name是存放数据的变量名,必须在声明光标签定义好。fetch...into 语句与select....into语句具有相同的意义

#将光标info_of_student中select语句查询出来的信息存入temp_name和temp_tel中,
其中temp_name和temp_tel已经定义好
fetch info_of_student into temp_name,temp_tel;

关闭光标(close cursor)

        光标使用完毕后,要及时关闭,在MySQL中采用close关键字关闭光标,语法格式如下:

close cursor_name;

        cursor_name:表示光标名称,例如:

#关闭info_of_student
close info_of_student;

注意:
        对于已经关闭的光标,在关闭后不能使用fetch来使用光标,光标在使用完后一定要关闭。

流程控制语句

        在MySQL中,常见的过程式SQL语句可以在一个存储过程体中,其中包括if语句、case语句、loop语句、while语句、iterate语句和leave语句,他们可以进行流程控制。

if语句

        if语句用来进行条件判断,根据不同的条件执行不同的操作。在执行时先判断if后的条件是否为真,若为真则执行then后面的语句,若为假则继续判断,语法格式如下:

if condition then

...

[elseif condition then]

.....

[else]

......

endif

USE `students`$$

DROP PROCEDURE IF EXISTS `example_if`$$

CREATE DEFINER=`root`@`localhost` PROCEDURE `example_if`(IN X INT)
BEGIN
	IF X=1 THEN
	SELECT 1;
	ELSEIF X=2 THEN
	SELECT 2;
	ELSE
	SELECT 3;
	END IF;

	END$$

DELIMITER ;
CALL example_if(1);

mysql存储过程打不开了 mysql存储过程cursor_存储过程_05

 case语句

        case语句为多分支语句结构,该语句首先从when后面的value中查找与case后面value相等的值,如果查找到则执行该分支的内容,若没找到,则执行else后面的内容,语法格式如下:

case value

when value then ......

[when value then......]

[else......]

end case;

        其中,value:表示条件判断的变量,when.... then中value表示变量的取值

DELIMITER $$

USE `students`$$

DROP PROCEDURE IF EXISTS `example_case`$$

CREATE DEFINER=`root`@`localhost` PROCEDURE `example_case`(IN X INT)
BEGIN
	CASE X
	WHEN 1 THEN SELECT 1;
	WHEN 2 THEN SELECT 2;
	ELSE SELECT 'else';
	 END CASE;

	END$$

DELIMITER ;
CALL example_case(3);

mysql存储过程打不开了 mysql存储过程cursor_mysql_06

while循环语句

        while循环语句执行时首先判断condition条件是否为真,如果为真则执行循环体,否则退出循环,语法格式如下:

while condition do

.....

end while;

#求1-100的和
DELIMITER $$

USE `students`$$

DROP PROCEDURE IF EXISTS `example_while`$$

CREATE DEFINER=`root`@`localhost` PROCEDURE `example_while`(OUT SUM INT)
BEGIN
	DECLARE i INT DEFAULT 1;
	DECLARE s INT DEFAULT 0;
	WHILE i<=100 DO
	SET s=s+i;
	SET i=i+1;
	END WHILE;
	SET SUM =s;
	

	END$$

DELIMITER ;
CALL example_while(@s);
SELECT @s;

mysql存储过程打不开了 mysql存储过程cursor_存储过程_07