您的位置:首页 > 数据库 > MySQL

MySQL存储过程之细节

2015-09-04 03:48 465 查看
1. ALTER and DROP

ALTER PROCEDURE p6 COMMENT 'Unfinished' //
DROP PROCEDURE p6 //


2. 与Oracle / SQL Server / DB2 / ANSI比较

  1) 与Oracle比较

    1> 摘要:

      a. Oracle允许在打开后再声明;MySQL必须在开始的时候使用

      b. Oracle允许"CURSOR cursorname IS"这样的声明方式;MySQL必须使用"DECLARE cursorname CURSOR"声明.

      c. Oracle不强制需要"()";MySQL必须有"()";

      d. Oracle允许在函数中访问表元素;MySQL不允许.

      e. Oracle支持"packages";MySQL不支持.

    2> 数据迁移的技巧

      把a:=b类似的赋值语句改成SET a=b

      将过程中的RETURN语句改为LEAVE <label_name>(label_name是为过程体定义的标号),如:      

/*在Oracle存储过程中*/
CREATE PROCEDURE
...
RETURN;
...
/*在MySQL存储过程中*/
CREATE PROCEDURE ()
label_at_start: BEGIN
...
LEAVE label_at_start;
END


      这一步仅在过程中需要,因为函数支持RETURN.

    3> 平行比较

OracleMySQL
CREATE PROCEDURECREATE PROCEDURE
sp_namesp_name
ASBEGIN
variable1 INTEGERDECLARE variable1 INTEGER
variable1 :=55SET variable1 = 55;
ENDEND
  2) 与SQL Server对比

    1> 摘要:

      a. SQL Server参数名字必须以@开关;MySQL参数名是常规标识符.

      b. SQL Server可以同时进行多个声明 DECLARE v1 [data type], v2 [data type]; ;MySQL每次只能声明一个 DECLARE v1 [data type]; DECLARE v2 [data type] .

      c. SQL Server存储过程体中没有BEGIN/END;MySQL必须有

      d. SQL Server中语句不需要分号结束;MySQL中除最后一句外必须有分号

      e. SQL Server可以进行SET NOCOUNT设置和IF @@ROWCOUNT判断;MySQL没有,但可以使用FOUND_ROWS()判断.

      f. SQL Server中循环使用WHILE...BEGIN语句;MySQL使用WHILE...DO语句

      g. SQL Server允许使用SELECT进行指派;MySQL只允许SET进行指派

      h. SQL Server允许在函数中访问表;MySQL不允许

      Microsoft SQL Server的区别特别多,所以将Microsoft或Sybase程序转换成MySQL程序将会是个冗长的过程,而且区别都是在语法定义上的,所以转换需要更多的特别的技巧.

    2>迁移技巧

      如果SQL Server的过程中有名为@xxx的变量,必须将其转换,因为@在MySQL中并不代表过程变量,而是全局变量;并且不要简单改成xxx,这可能会和数据库中某个表的某个字段冲突.因此可以把@替换成你的自定义字符串前缀,如var_xxx.

    3> 平行对比

SQL ServerMySQL
CREATE PROCEDURECREATE PROCEDURE
sp_procedure1sp_procedure1()
ASBEGIN
DECLARE @x VARCHAR(100)DECLARE var_x VARCHAR(100);
EXECUTE sp_procedure2 @xCALL sp_procedure2(var_x);
DECLARE c CURSOR FORDECLARE c CURSOR FOR
SELECT * FROM tSELECT * FROM t;
ENDEND
  3) 与DB2比较

    1>摘要

      a. DB2允许PATH和SIGAL语句并允许函数访问表;MySQL不允许

      b. DB2允许过程名重载;MySQL不允许

      c. DB2有"label_x:...GOTO label_x"语法;MySQL有非正式的"label_x:...GOTO label_x"语法

      DB2存储过程基本和MySQL一致,唯一的不同在于MySQL还没有引进DB2的一些语句,还有就是DB2允许重载,因此DB2可以有两个名字一样的存储过程,只通过参数或返回类型来决定执行哪个所以DB2存储过程可以向下与MySQL的兼容.

    2> 迁移技巧

      此处迁移基本不需要任何技巧,MySQL没有SIGNAL语句,可以在其他地方讨论临时工作区的问题,而对DB2的GOTO语句,可直接用MySQL中的GOTO语句直接代替.PATH(用于确定DBMS寻找过程的数据库目录)问题只需要在过程名前加上前缀就可以避免,关于函数访问表的问题可用带OUT参数的存储过程代替.

    3>平行对比

DB2MySQL
CREATE PROCEDURECREATE PROCEDURE
sp_namesp_name
(parameter1 INTEGER)(parameter1 INTEGER)
LANGUAGE SQLLANGUAGE SQL
BEGINBEGIN
  DECLARE v INTEGER;  DECLARE v INTEGER;
  IF parameter1 >= 5 THEN  IF parameter1 >= 5 THEN
    CALL p26();    CALL p26();
    SET v = 2;    SET v = 2;
  END IF;  END IF;
  INSERT INTO t VALUES ( v );  INSERT INTO t VALUES ( v );
END @END //
  4) 与sql标准的比较

    标准sql的要求:(跟DB2中的一样)

    MySQL的目标是支持以下两个标准SQL特性:存储模式和计算完整性.DB2与MySQL相似的原因是两者都支持标准SQL中的存储过程.因此,MySQL和DB2的区别就像我们背离ANSI/ISO标准语法那样,但比Oracle或SQL Server更标准.

3. 编程风格

CREATE PROCEDURE p ()
BEGIN
/* Use comments! */
UPDATE t SET s1 = 5;
CREATE TRIGGER t2_ai ...
END;//


  上例中的编程风格:关键字大写,命名约定中,表名为t,列名为s...

  注释和C语言中的一样,在BEGIN后缩进(一般是一个TAB字符);在END前回缩

4. 几个例子

  1)字符串连接函数 -- tables_concat():

CREATE PROCEDURE tables_concat  (OUT parameter1 VARCHAR(1000))
BEGIN
DECLARE variable2 CHAR(100);
DECLARE c CURSOR FOR SELECT table_name FROM information_schema.tables;
DECLARE EXIT HANDLER FOR NOT FOUND BEGIN END;    /* 1 */
SET sql_mode='ansi';                             /* 2 */
SET parameter1 = '';
OPEN c;
LOOP
FETCH c INTO variable2;                        /* 3 */
SET parameter1 = parameter1 || variable2 || '.';
END LOOP;
CLOSE c;
END;
/* 1:  这里的"BEGIN END"语句没有任何作用,就像其他DBMS中的NULL语句。 */
/* 2:  将sql_mode设置为'ansi'以便"||"能正常连接,在退出存储过程后sql_mode仍为'ansi'。  */
/* 3:  另一种跳出循环LOOP的方法:声明EXIT出错处理,当FETCH没有返回行时。 */


  这是所有表名连接到一个单一字符串的函数,可以和MySQL内建的group_concat()函数对比一下.以下是我调用该过程的示例和结果:

mysql> CALL tables_concat(@x);
/*Query OK, 0 rows affected (0.05 sec)    */
mysql> SELECT @x;
/* SCHEMATA.TABLES.COLUMNS.CHARACTER_SETS.COLLATIONS.C  OLLATION_CHARACTER_SET_APPLICABILITY.ROUTINES.STATIST  ICS.VIEWS.USER_PRIVILEGES.SCHEMA_PRIVILEGES.TABLE_PRI   VILEGES.COLUMN_PRIVILEGES.TABLE_CONSTRAINTS.KEY_COLUM  N_USAGE.TABLE_NAMES.columns_priv.db.fn.func.help_cate  gory.help_keyword.help_relation.help_topic.host.proc.  tables_priv.time_zone.time_zone_leap_second.time_zone  1 row in set (0.00 sec) */


  下面示例获得符合条件行数,类似其他DBMS中的ROWNUM():

CREATE FUNCTION rno ()
RETURNS INT
BEGIN
SET @rno = @rno + 1;
RETURN @rno;
END;


  使用示例如下:

mysql> SET @rno = 0;//
/*Query OK, 0 rows affected (0.00 sec)*/
mysql> SELECT rno(),s1,s2 FROM t;//
/+-------+------+------+
| rno() |  s1  |  s2  |
/+-------+------+------+
|   1   |   1  |   a  |
|   2   |   2  |   b  |
|   3   |   3  |   c  |
|   4   |   4  |   d  |
|   5   |   5  |   e  |
/+-------+------+------+
5 rows in set (0.00 sec)*/


  2) running_total()

    这个累加的函数建立在;rno()基础上,不同之处在于我们要在每次调用时传值到参数中:

CREATE FUNCTION running_total (IN adder INT)
RETURNS INT
BEGIN
SET @running_total = @running_total + adder;
RETURN @running_total;
END;
/* 下面是函数调用级结果 */
mysql> SET @running_total = 0;//
/*  Query OK, 0 rows affected (0.01 sec)    */
mysql> SELECT s1,running_total(s1),s2 FROM t ORDER BY s1;//
/*
+------+-------------------+------+
| s1   | running_total(s1) | s2   |
+------+-------------------+------+
|    1 |                 1 | a    |
|    2 |                 3 | b    |
|    3 |                 6 | c    |
|    4 |                10 | d    |
|    5 |                15 | e    |
+------+-------------------+------+
5 rows in set (0.01 sec)
*/


  3) MyISAM外键插入

    MyISAM存储引擎不支持外键,但是你可以将这个逻辑加入存储过程引擎进行检查:

CREATE PROCEDURE fk_insert (p_fk INT, p_animal VARCHAR(10))
BEGIN
DECLARE v INT;
BEGIN
DECLARE EXIT HANDLER FOR SQLEXCEPTION, NOT FOUND
    SET v = 0;
IF p_fk IS NOT NULL THEN
SELECT 1 INTO v FROM tpk WHERE cpk = p_fk LIMIT 1;
INSERT INTO tfk VALUES (p_fk, p_animal);
ELSE
SET v = 1;
END IF;
END;
IF v <> 1 THEN
DROP TABLE `The insertion failed`;
END IF;
END;


    注意:SQLEXCEPTION或NOT FOUND条件都会导致v变0,而如果这些条件为假,则v会变成1,因为SELECT会给v赋值1,而EXIT HANDLER没有运行.以下看看运行结果:

mysql> CREATE TABLE tpk (cpk INT PRIMARY KEY);//
/* Query OK, 0 rows affected (0.01 sec)  */
mysql> CREATE TABLE tfk (cfk INT, canimal VARCHAR(10));//
/* Query OK, 0 rows affected (0.00 sec)    */
mysql> INSERT INTO tpk VALUES (1),(7),(10);//
/* Query OK, 3 rows affected (0.01 sec)  Records: 3  Duplicates: 0  Warnings: 0    */
mysql> CALL fk_insert(1,'wombat');//
/* Query OK, 1 row affected (0.02 sec)    */
mysql> CALL fk_insert(NULL,'wallaby');//
/* Query OK, 0 rows affected (0.00 sec)    */
mysql> CALL fk_insert(17,'wendigo');//
/* ERROR 1051 (42S02): Unknown table 'The insertion failed' */


  4) 错误传递

    如果过程1调用过程2,过程2调用过程3,过程3中的错误就会传递到过程1.如果没有异常处理器捕获异常,那异常就会传递,导致过程2出错,进而最后导致过程1出错,最终异常传递到了调用者(MySQL客户端实例).这种特性使得标准SQL中存在SIGNAL语句来使异常强制发生,其他DBMS中也有类似措施(RAISEERROR).MySQL还不支持SIGNAL,直到支持此特性出来之前,可以用下面的异常处理方式:

CREATE PROCEDURE procedure1 ()
BEGIN
CALL procedure2();
SET @x = 1;
END;
CREATE PROCEDURE procedure2 ()
BEGIN
CALL procedure3();
SET @x = 2;
END;
CREATE PROCEDURE procedure3 ()
BEGIN
DROP TABLE error.`error #7815`;
SET @x = 3;
END;
/* 调用过程1后结果如下:  */
mysql> CALL procedure1()//
/*ERROR 1051 (42S02): Unknown table 'error #7815' */


    @x并没有改变,因为没有一条"SET @x = ..."语句成功被执行,而使用DROP可以产生一些可供诊断的错误信息.

  5) 库

    对库的应用有详细的规格说明,为使拥有权限的用户都能调用过程,可以如下设置: GRANT ALL ON database-name.* TO user-name; 如果要其他用户只有访问过程的权限,只要定义SQL SECURITY DEFINER特性就可以了,而这个选项是默认的,但最好显式的声明出来.

    下面是一个向数据库中添加书本的过程,这里必须测试书的id是否确定,书名是否为空.例子是对MySQL不支持的CHECK限制功能的替代.

CREATE PROCEDURE add_book
(p_book_id INT, p_book_title VARCHAR(100))
SQL SECURITY DEFINER
BEGIN
IF p_book_id < 0 OR p_book_title='' THEN
SELECT 'Warning:
Bad parameters';
END IF;
INSERT INTO books VALUES (p_book_id, p_book_title);
END;


    我们需要一个添加买主的过程,过程必须检查是否有超过一个的买主,如有则给出警告.这个可以在一个子查询中完成:

IF (SELECT COUNT(*) FROM table-name) > 2) THEN ... END IF;


    不过,目前子查询功能有漏洞,于是可用"SELECT COUNT(*) INTO variable-name"代替.

CREATE PROCEDURE add_patron
(p_patron_id INT, p_patron_name VARCHAR(100))
SQL SECURITY DEFINER
BEGIN
DECLARE v INT DEFAULT 0;
SELECT COUNT(*) FROM patrons INTO v;
IF v > 2 THEN
SELECT 'warning: already there are ',v,'patrons!';
END IF;
INSERT INTO patrons VALUES (p_patron_id,p_patron_name);
END;


    下面需要书本付帐的过程,在事务处理过程中我们希望显示已经拥有本书的买主,及其拥有的书,这些信息可以通过对游标CURSOR的Fetch来获得,可以有两种不同的方法来测试是否fetch数据已完毕:检查变量在fetch动作后是否为NULL;通过NOT FOUND错误处理捕获fetch的失败动作.

CREATE PROCEDURE checkout (p_patron_id INT, p_book_id INT)
SQL SECURITY DEFINER
BEGIN
DECLARE v_patron_id, v_book_id INT;
DECLARE no_more BOOLEAN default FALSE;
DECLARE CONTINUE HANDLER FOR NOT FOUND SET no_more=TRUE;
BEGIN
DECLARE c1 CURSOR FOR SELECT patron_id FROM transactions WHERE book_id = p_book_id;
OPEN c1;
SET v_patron_id=NULL;
FETCH c1 INTO v_patron_id;
IF v_patron_id IS NOT NULL THEN
SELECT 'Book is already out to this patron:', v_patron_id;
END IF;
CLOSE c1;
END;
BEGIN
DECLARE c2 CURSOR FOR SELECT book_id FROM transactions WHERE patron_id = p_patron_id;
OPEN c2;
book_loop: LOOP
FETCH c2 INTO v_book_id;
IF no_more THEN
LEAVE book_loop;
END IF;
SELECT 'Patron already has this book:', v_book_id;
END LOOP;
END;
INSERT INTO transactions VALUES (p_patron_id, p_book_id);    END;


  6) 分层次

    hierarchy()过程实现的是其他DBMS中CONNECT BY部分功能

CREATE PROCEDURE hierarchy (start_with CHAR(10))
proc:BEGIN
DECLARE temporary_table_exists BOOLEAN;
BEGIN
DECLARE CONTINUE HANDLER FOR SQLEXCEPTION
BEGIN
END;
DROP TABLE IF EXISTS Temporary_Table;
END;
BEGIN
DECLARE v_person_id, v_father_id INT;
DECLARE v_person_name CHAR(20);
DECLARE done, error BOOLEAN DEFAULT FALSE;
DECLARE CONTINUE HANDLER FOR NOT FOUND SET done = TRUE;
DECLARE CONTINUE HANDLER FOR SQLEXCEPTION
SET error = TRUE;

CREATE TEMPORARY TABLE Temporary_Table (person_id INT, person_name CHAR(20), father_id INT);
IF error THEN
SELECT 'CREATE TEMPORARY failed';
LEAVE proc;
END IF;
SET temporary_table_exists=TRUE;
SELECT person_id, person_name INTO v_person_id, v_person_name FROM Persons WHERE person_name = start_with limit 1;
IF error THEN
SELECT 'First SELECT failed';
LEAVE proc;
END IF;
IF v_person_id IS NOT NULL THEN
INSERT INTO Temporary_Table VALUES (v_person_id, v_person_name, v_father_id);
IF error THEN
SELECT 'First INSERT failed'; LEAVE proc; END IF;
CALL hierarchy2(v_person_id);
IF error THEN
SELECT 'First CALL hierarchy2() failed';
END IF;
END IF;
SELECT person_id, person_name, father_id FROM Temporary_Table;
IF error THEN
SELECT 'Temporary SELECT failed';
LEAVE proc;
END IF;
END;
IF temporary_table_exists THEN
DROP TEMPORARY TABLE Temporary_Table;
END IF;
END;  CREATE PROCEDURE hierarchy2 (start_with INT)


proc:BEGIN
DECLARE v_person_id INT, v_father_id INT;
DECLARE v_person_name CHAR(20);
DECLARE done, error BOOLEAN DEFAULT FALSE;
DECLARE c CURSOR FOR SELECT person_id, person_name, father_id FROM Persons WHERE father_id = start_with;
DECLARE CONTINUE HANDLER FOR NOT FOUND SET done = TRUE;
DECLARE CONTINUE HANDLER FOR SQLEXCEPTION SET error = TRUE;
OPEN c;
IF error THEN
SELECT 'OPEN failed';
LEAVE proc;
END IF;
REPEAT
SET v_person_id=NULL;
FETCH c INTO v_person_id, v_person_name, v_father_id;
IF error THEN
SELECT 'FETCH failed';
LEAVE proc;
END IF;
IF done=FALSE THEN
INSERT INTO Temporary_Table VALUES (v_person_id, v_person_name, v_father_id);
IF error THEN
SELECT 'INSERT in hierarchy2() failed';
END IF;
CALL hierarchy2(v_person_id);
IF error THEN
SELECT 'Recursive CALL hierarchy2() failed';
END IF;
END IF;
UNTIL done = TRUE
END REPEAT;
CLOSE c;
IF error THEN
SELECT 'CLOSE failed';
END IF;
END;


  下是调用hierarchy()后的结果:

mysql> CREATE TABLE Persons (person_id INT, person_name CHAR(20), father_id INT);//
/*  Query OK, 0 rows affected (0.00 sec)    */
mysql> INSERT INTO Persons VALUES (1,'Grandpa',NULL);//
/*  Query OK, 1 row affected (0.00 sec)*/
mysql> INSERT INTO Persons VALUES (2,'Pa-1',1),(3,'Pa-2',1);//
/*  Query OK, 2 rows affected (0.00 sec)  Records: 2  Duplicates: 0  Warnings: 0     */
mysql> INSERT INTO Persons VALUES (4,'Grandson-1',2),(5,'Grandson-2',2);//
/*  Query OK, 2 rows affected (0.00 sec)  Records: 2  Duplicates: 0  Warnings: 0     */
mysql> call hierarchy('Grandpa')//
/*
+-----------+-------------+-----------+
| person_id | person_name | father_id |
+-----------+-------------+-----------+
|         1 | Grandpa     |      NULL |
|         2 | Pa-1        |         1 |
|         4 | Grandson-1  |         2 |
|         5 | Grandson-2  |         2 |
|         3 | Pa-2        |         1 |
+-----------+-------------+-----------+
5 rows in set (0.01 sec)
Query OK, 0 rows affected (0.01 sec)
*/
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: