mysql 存储过程示例


解读存储过程

# DECLARE语句是在复合语句中声明变量的指令

# DATEDIFF() 函数返回两个日期之间的天数。

脚本中有多行,我们希望不按照分号来一句一句执行,希望一段代码作为一个整体执行,在脚本最前面加上 delimiter $$

DECLARE rs_cursor CURSOR FOR,游标(cursor)必须在声明处理程序之前被声明,并且变量和条件必须在声明游标或处理程序之前被声明。


代码如下:

DELIMITER $$ 
DROP PROCEDURE IF EXISTS getUserInfo $$
CREATE PROCEDURE getUserInfo(in date_day datetime)
-- 
-- 实例
-- 存储过程名为:getUserInfo
-- 参数为:date_day日期格式:2008-03-08
--
    BEGIN
declare _userName varchar(12); -- 用户名
declare _chinese int ; -- 语文
declare _math int ;    -- 数学
declare done int;
-- 定义游标
DECLARE rs_cursor CURSOR FOR SELECT username,chinese,math from userInfo where datediff(createDate, date_day)=0;
DECLARE CONTINUE HANDLER FOR NOT FOUND SET done=1;
-- 获取昨天的日期
if date_day is null then
   set date_day = date_add(now(),interval -1 day);
end if;
open rs_cursor; 
cursor_loop:loop
   FETCH rs_cursor into _userName, _chinese, _math; -- 取数据

   if done=1 then
    leave cursor_loop;
   end if;
   -- 更新表
   update infoSum set total=_chinese+_math where UserName=_userName;
end loop cursor_loop;
close rs_cursor;

    END$$
DELIMITER ;
 



-------完整事例---------------------------

mysql被oracle收购后,从mysql-5.5开始,将InnoDB作为默认存储引擎,是一次比较重大的突破。InnoDB作为支持事务的存储引擎,拥有相关的RDBMS特性:包括ACID事务支持,数据完整性(外健),灾难恢复能力等特性。

使用mysql做为数据库的话,将来程序员肯定要写很多,存储过程,function等。在写些东东的时候,游标肯定是少不了的。下面简单简介一下。
一,什么是游标(cursor)

个人觉得就是一个cursor,就是一个标识,用来标识数据取到什么地方了。你也可以把它理解成数组中的下标。

二,游标(cursor)的特性

1,只读的,不能更新的。
2,不滚动的
3,不敏感的,不敏感意为服务器可以活不可以复制它的结果表

游标(cursor)必须在声明处理程序之前被声明,并且变量和条件必须在声明游标或处理程序之前被声明。

三,使用游标(cursor)

1.声明游标

DECLARE cursor_name CURSOR FOR select_statement
这个语句声明一个游标。也可以在子程序中定义多个游标,但是一个块中的每一个游标必须有唯一的名字。声明游标后也是单条操作的,但是不能用SELECT语句不能有INTO子句。

2. 游标OPEN语句

OPEN cursor_name
这个语句打开先前声明的游标。

3. 游标FETCH语句

FETCH cursor_name INTO var_name [, var_name] ...
这个语句用指定的打开游标读取下一行(如果有下一行的话),并且前进游标指针。

4. 游标CLOSE语句

CLOSE cursor_name
这个语句关闭先前打开的游标。

四,应用举例

1,测试表和数据

查看 复制 打印 ?
  1. mysql> show create table users\G;      //创建一个测试表  
  2. *************************** 1. row ***************************  
  3.  Table: users  
  4. Create Table: CREATE TABLE `users` (  
  5.  `ID` bigint(20) unsigned NOT NULL AUTO_INCREMENT,  
  6.  `user_name` varchar(60) NOT NULL DEFAULT '',  
  7.  `user_pass` varchar(64) NOT NULL DEFAULT '',  
  8.  PRIMARY KEY (`ID`)  
  9. ) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8  
  10. 1 row in set (0.00 sec)  
  11.   
  12. mysql> select * from users;           //测试数据  
  13. +----+-----------+-----------+  
  14. | ID | user_name | user_pass |  
  15. +----+-----------+-----------+  
  16. |  1 | tank      | tank      |  
  17. |  2 | zhang     | zhang     |  
  18. |  3 | ying      | ying      |  
  19. |  4 | tank      | zhang     |  
  20. +----+-----------+-----------+  
  21. 4 rows in set (0.00 sec)  

2,测试存储过程

查看 复制 打印 ?
  1. mysql> delimiter |  
  2. mysql> create procedure test_cursor (in param int(10),out result varchar(90))  
  3.  -> begin  
  4.  ->     declare name varchar(20);  
  5.  ->     declare pass varchar(20);  
  6.  ->     declare done int;  
  7.  ->     declare cur_test CURSOR for select user_name,user_pass from test.users;  
  8.  ->     declare continue handler FOR SQLSTATE '02000' SET done = 1;  
  9.  ->     if param then  
  10.  ->         select concat_ws(',',user_name,user_pass) into result from test.users where id=param;  
  11.  ->     else  
  12.  ->         open cur_test;  
  13.  ->         repeat  
  14.  ->             fetch cur_test into name, pass;  
  15.  ->             select concat_ws(',',result,name,pass) into result;  
  16.  ->         until done end repeat;  
  17.  ->         close cur_test;  
  18.  ->     end if;  
  19.  -> end;|  
  20. Query OK, 0 rows affected (0.00 sec)  

注意,在命令行缩进时,不要用tab,不然会提示,

Display all 749 possibilities? (y or n)
? MBRINTERSECTS
ABS MBROVERLAPS

。。。 。。。。。。。

1行,创建一个存储过程,注意:如果我把out result varchar(90)改成out result varchar,返回的结果中只有一个字符。
2行,开始
3行,定义一个变量name
4行,定义变量pass
5行,定义一下结束标识
6行,定义一个光标       注意:declare 的内容不要放到if里面,不然会报错误的。
7行,如果sqlstate等于02000时,把done设置成1,也就是找不到数据时
8,10,17行,if判断
9行,根据参数,把数据取出来,放到result中
11行,打开光标
12,15行,repeat循环,根php的do while原理一样
13行,从光标中取出数据。
14行,将数据合并起来
16行,关闭光标
17,18行,标签闭合。

3,测试结果

查看 复制 打印 ?
  1. mysql> call test_cursor(3,@test);  
  2. Query OK, 0 rows affected (0.00 sec)  
  3.   
  4. mysql> select @test;                             //这里很像php中的,传引用  
  5. +-----------+  
  6. | @test     |  
  7. +-----------+  
  8. | ying,ying |  
  9. +-----------+  
  10. 1 row in set (0.00 sec)  
  11.   
  12. mysql> call test_cursor('',@test);  
  13. Query OK, 0 rows affected, 1 warning (0.00 sec)  
  14.   
  15. mysql> select @test;  
  16. +-------------------------------------------------------+  
  17. | @test                                                 |  
  18. +-------------------------------------------------------+  
  19. | tank,tank,zhang,zhang,ying,ying,tank,zhang,tank,zhang |  
  20. +-------------------------------------------------------+  
  21. 1 row in set (0.00 sec)  

(完)














你可能感兴趣的:(mysql 存储过程示例)