【MySQL筆記】觸發器,存儲過程和函數


一、觸發器

觸發器(TRIGGER):是由事件來觸發某個操作。當數據庫系統執行這些事件時,就會激活觸發器執行相應的操作。MySQL從5.0.2版本開始支持觸發器。
觸發事件:INSERT語句、UPDATE語句和DELETE語句。

1、創建觸發器

1.1、創建只有一個執行語句的觸發器

CREATE  TRIGGER 觸發器名  BEFORE | AFTER  觸發事件
ON  表名 FOR  EACH  ROW  執行語句
 

1.2、創建有多個執行語句的觸發器

DELIMITER&&
CREATE  TRIGGER 觸發器名  BEFORE | AFTER  觸發事件
ON  表名 FOR  EACH  ROW
BEGIN
執行語句列表
END
&&
DELEMITER ;
 
注: DELIMITER,重定義結束符,分隔符。一般SQL “;”結束,在創建多個語句執行的觸發器時,要用到“;”,所以用DELIMETER來切換一下。

1.3例: 

CREATE TRIGGER dept_tig1 BEFORE INSERT ON department
FOR EACH ROW
INSERT INTO trigger_time VALUES(NOWS());

1.4觸發器的使用

 MySQL中,觸發器執行的順序是BEFORE觸發器、表操作(INSERT、UPDATE 和DELETE)、AFTER觸發器觸發器中不能包含START TRANSACTION(事務,並發控制單位), COMMIT,ROLLBACK,CALL等。

2、查看觸發器

SHOW  TRIGGERS ;

SELECT  *  FROM  information_schema. triggers ;
SELECT * FROM information_schema.triggers WHERE TRIGGER_NAME='觸發器名';

注:information_schema.triggers:表示information_schema數據庫下面的triggers表。

3、刪除觸發器

DROP TRIGGER  觸發器名 ;

教程鏈接: http://blog.csdn.net/kimsoft/article/details/6757457

二、存儲過程和函數

存儲過程(Stored Procedure),是一組為了完成特定功能的SQL 語句,集經編譯后存儲在數據庫中,用戶通過指定存儲過程的名字並給出參數,如果該存儲過程帶有參數來執行。

1、MySQL存儲過程的創建

1.1格式

CREATE PROCEDURE 過程名 ([過程參數[,...]])
[特性 ...] 過程體

1.2例: 

  1. mysql> DELIMITER //  
  2. mysql> CREATE PROCEDURE proc1(OUT int)  
  3.     -> BEGIN 
  4.     -> SELECT COUNT(*) INTO FROM user;  
  5.     -> END 
  6.     -> //  
  7. mysql> DELIMITER ;

注:
(1)這里需要注意的是DELIMITER //和DELIMITER ;兩句,DELIMITER是分割符的意思,因為MySQL默認以";"為分隔符,如果我們沒有聲明分割符,那么編譯器會把存儲過程當成SQL語句進行處理,則存儲過程的編譯過程會報錯,所以要事先用DELIMITER關鍵字申明當前段分隔符,這樣MySQL才會將";"當做存儲過程中的代碼,不會執行這些代碼,用完了之后要把分隔符還原。
(2)存儲過程根據需要可能會有輸入、輸出、輸入輸出參數, 這里有一個輸出參數s,類型是int型,如果有多個參數用","分割開。
(3)過程體的開始與結束使用BEGIN與END進行標識。
(4)MySQL不支持Select Into語句直接備份表結構和數據,不支持:Select * Into new_table_name from old_table_name;
    替換方法1:
          create table dust select * from student;//用於復制前未創建新表dust的情況下
    替換方法2:
          insert into dust select * from student;//已經創建了新表dust的情況下

1.3參數

MySQL存儲過程的參數用在存儲過程的定義,共有三種參數類型,IN,OUT,INOUT,形式如:
CREATE PROCEDURE  過程名 ( [   [IN |OUT |INOUT ] 參數名 數據類形...  ] )

  • IN 輸入參數:表示該參數的值必須在調用存儲過程時指定,在存儲過程中修改該參數的值不能被返回,為默認值
  • OUT 輸出參數:該值可在存儲過程內部被改變,並可返回
  • INOUT 輸入輸出參數:調用時指定,並且可被改變和返回

1) IN參數例子
  1. mysql DELIMITER //  
  2. mysql CREATE PROCEDURE demo_in_parameter(IN p_in int)  
  3. -> BEGIN   
  4. -> SELECT p_in;   
  5. -> SET p_in=2;   
  6. -> SELECT p_in;   
  7. -> END  
  8. -> //  
  9. mysql DELIMITER 


執行結果:

  1. mysql SET @p_in=1;  
  2. mysql CALL demo_in_parameter(@p_in);  
  3. +------+  
  4. p_in |  
  5. +------+  
  6.      
  7. +------+  
  8.  
  9. +------+  
  10. p_in |  
  11. +------+  
  12.      
  13. +------+  
  14.  
  15. mysql> SELECT @p_in;  
  16. +-------+  
  17. @p_in |  
  18. +-------+  
  19.     |  
  20. +-------+  

以上可以看出,p_in雖然在存儲過程中被修改,但並不影響@p_id的值


2)OUT參數例子
  1. mysql DELIMITER //  
  2. mysql CREATE PROCEDURE demo_out_parameter(OUT p_out int)  
  3. -> BEGIN 
  4. -> SELECT p_out;  
  5. -> SET p_out=2;  
  6. -> SELECT p_out;  
  7. -> END;  
  8. -> //  
  9. mysql DELIMITER 


執行結果
:

  1. mysql SET @p_out=1;  
  2. mysql CALL sp_demo_out_parameter(@p_out);  
  3. +-------+  
  4. p_out   
  5. +-------+  
  6. NULL    
  7. +-------+  
  8.  
  9. +-------+  
  10. p_out |  
  11. +-------+  
  12.       
  13. +-------+  
  14.  
  15. mysql> SELECT @p_out;  
  16. +-------+  
  17. p_out |  
  18. +-------+  
  19.     |  
  20. +-------+  

3)
  1. mysql DELIMITER //   
  2. mysql CREATE PROCEDURE demo_inout_parameter(INOUT p_inout int)   
  3. -> BEGIN 
  4. -> SELECT p_inout;  
  5. -> SET p_inout=2;  
  6. -> SELECT p_inout;   
  7. -> END;  
  8. -> //   
  9. mysql DELIMITER 

 

 

執行結果 :
  1. mysql > SET @p_inout=1;  
  2. mysql > CALL demo_inout_parameter(@p_inout) ;  
  3. +---------+  
  4. p_inout |  
  5. +---------+  
  6.       |  
  7. +---------+  
  8.  
  9. +---------+  
  10. p_inout   
  11. +---------+  
  12.       |  
  13. +---------+  
  14.  
  15. mysql > SELECT @p_inout;  
  16. +----------+  
  17. @p_inout   
  18. +----------+  
  19.        |  
  20. +----------+ 

1.4變量

1)定義變量
DECLARE variable_name [,variable_name...] datatype [DEFAULT value];
注:datatype為MySQL的數據類型,如:int, float, date, varchar(length)

例如:

  1. DECLARE l_int int unsigned default 4000000;  
  2. DECLARE l_numeric number(8,2) DEFAULT 9.95;  
  3. DECLARE l_date date DEFAULT '1999-12-31';  
  4. DECLARE l_datetime datetime DEFAULT '1999-12-31 23:59:59';  
  5. DECLARE l_varchar varchar(255) DEFAULT 'This will not be padded'
2)變量賦值
SET 變量名 = 表達式值 [,variable_name = expression ...]
3)用戶變量

  • MySQL客戶端使用用戶變量

  1. mysql SELECT 'Hello World' into @x;  
  2. mysql SELECT @x;  
  3. +-------------+  
  4.   @x        |  
  5. +-------------+  
  6. Hello World |  
  7. +-------------+  
  8. mysql SET @y='Goodbye Cruel World';  
  9. mysql SELECT @y;  
  10. +---------------------+  
  11.     @y              |  
  12. +---------------------+  
  13. Goodbye Cruel World |  
  14. +---------------------+  
  15.  
  16. mysql SET @z=1+2+3;  
  17. mysql SELECT @z;  
  18. +------+  
  19. @z   |  
  20. +------+  
  21.    |  
  22. +------+  

  • 在存儲過程中使用用戶變量

  1. mysql CREATE PROCEDURE GreetWorld( SELECT CONCAT(@greeting,World');  
  2. mysql SET @greeting='Hello';  
  3. mysql CALL GreetWorld( );  
  4. +----------------------------+  
  5. CONCAT(@greeting,World'|  
  6. +----------------------------+  
  7.  Hello World               |  
  8. +----------------------------+  

 

  • 在存儲過程間傳遞全局范圍的用戶變量
  1. mysql> CREATE PROCEDURE p1()   SET @last_procedure='p1';  
  2. mysql> CREATE PROCEDURE p2() SELECT CONCAT('Last procedure was ',@last_proc);  
  3. mysql> CALL p1( );  
  4. mysql> CALL p2( );  
  5. +-----------------------------------------------+  
  6. CONCAT('Last procedure was ',@last_proc  |  
  7. +-----------------------------------------------+  
  8. Last procedure was p1                         |  
  9. +-----------------------------------------------+  
注:
1) 用戶變量名一般以@開頭, 濫用用戶變量會導致程序難以理解及管理。
2) CONCAT(str1,str2,…) 連接函數
返回結果為連接參數產生的字符串。如有任何一個參數為NULL ,則返回值為 NULL。
例:
mysql> select concat('11','22','33');
+------------------------+
| concat('11','22','33') |
+------------------------+
| 112233 |
+------------------------+

4)變量作用域

        內部的變量在其作用域范圍內享有更高的優先權,當執行到end。變量時,內部變量消失,此時已經在其作用域外,變量不再可見了,此時在存儲

過程外再也不能找到這個申明的變量,但是你可以通過out參數或者將其值指派給會話變量來保存其值。

例: 

  1. mysql DELIMITER //  
  2. mysql CREATE PROCEDURE proc3()  
  3.      -> begin 
  4.      -> declare x1 varchar(5) default 'outer';  
  5.      -> begin 
  6.      -> declare x1 varchar(5) default 'inner';  
  7.      -> select x1;  
  8.      -> end;  
  9.      -> select x1;  
  10.      -> end;  
  11.      -> //  
  12. mysql DELIMITER ;  

1.5注釋

MySQL存儲過程可使用兩種風格的注釋
雙模杠(--): 該風格一般用於單行注釋
?c風格:一般用於多行注釋
例:

 mysql DELIMITER //  

  1. mysql CREATE PROCEDURE proc1 --name存儲過程名  
  2. -> (IN parameter1 INTEGER  
  3. -> BEGIN   
  4. -> DECLARE variable1 CHAR(10);   
  5. -> IF parameter1 17 THEN   
  6. -> SET variable1 'birds'  
  7. -> ELSE 
  8. -> SET variable1 'beasts'  
  9. -> END IF;   
  10. -> INSERT INTO table1 VALUES (variable1);  
  11. -> END   
  12. -> //  
  13. mysql DELIMITER 

2、存儲函數

存儲函數只支持輸入參數,並且輸入參數前沒有IN或INOUT,返回參數是在函數體中使用return返回要返回的數據變量的形式來表示的。

2.1格式:

create function 函數([函數參數[,….]]) Returns 返回類型
  Begin
    If(
      Return (返回的數據)
    Else
      Return (返回的數據)
    end if;
  end;

2.2例:

          create function count_news(hits int) returns int

3、調用存儲過程和函數

  • 存儲過程是通過CALL語句來調用的。調用存儲過程:
               CALL  sp_name([parameter[,…]]) ;
  • 存儲函數的使用方法與MySQL內部函數的使用方法是一樣的,直接調用。

  • 執行存儲過程和存儲函數需要擁有EXECUTE權限。EXECUTE權限的信息存儲在information_schema數據庫下面的USER_PRIVILEGES表中

4、查看存儲過程和函數

4.1查看存儲過程       

          查看一個數據庫下面有那些表,我們一般采用show tables;進行查看。那么我們要查看某個數據庫下面的存儲過程,是否也可以采用呢?答案是,我們可以查看某個數據庫下面的存儲過程:
          SHOW { PROCEDURE| FUNCTION } STATUS [ LIKE  ' pattern ' ];

例:
select name from mysql.proc where db=’數據庫名’;

select routine_name from information_schema.routines where routine_schema='數據庫名';

show procedure status where db='數據庫名';

4.2查看詳細的存儲過程

         如果我們想知道,某個存儲過程的詳細,那我們又該怎么做呢?是不是也可以像操作表一樣用describe 表名進行查看呢? 答案是:我們可以用一下方法查看存儲過程的詳細:
          SHOW CREATE PROCEDURE 數據庫.存儲過程名;
就可以查看當前存儲過程的詳細。

例:
SHOW CREATE {PROCEDURE | FUNCTION } sp_name ;
SELECT * FROMinformation_schema.Routines WHERE ROUTINE_NAME=' sp_name ' ;

4.3、修改存儲過程和函數

ALTER PROCEDURE: 更改用CREATE PROCEDURE 建立的預先指定的存儲過程,其不會影響相關存儲過程或存儲功能。
格式:
ALTER {PROCEDURE| FUNCTION} sp_name [characteristic ...]
characteristic:
{ CONTAINS SQL |NO SQL | READS SQL DATA | MODIFIES SQL DATA }
| SQL SECURITY {DEFINER | INVOKER }
| COMMENT'string'

4.4、刪除存儲過程和函數

DROP {PROCEDURE| FUNCTION } sp_name;

4.5存儲函數與存儲過程的區別

1)存儲函數有且只有一個返回值,而存儲過程不能有返回值。
2) 函數只能有輸入參數,而且不能帶in, 而存儲過程可以有多個in,out,inout參數。
3 )存儲過程中的語句功能更強大,存儲過程可以實現很復雜的業務邏輯,而函數有很多限制,如不能在函數中使用insert,update,delete,create等語句;存儲函數只完成查詢的工作,可接受輸入參數並返回一個結果,也就是函數實現的功能針對性比較強。
4) 存儲過程可以調用存儲函數。但函數不能調用存儲過程。
5) 存儲過程一般是作為一個獨立的部分來執行(call調用)。而函數可以作為查詢語句的一個部分來調用。

4.6MySQL存儲過程的控制語句

4.6.1條件語句

1)if-then -else語句
  1. mysql DELIMITER //  
  2. mysql CREATE PROCEDURE proc2(IN parameter int)  
  3.      -> begin 
  4.      -> declare var int;  
  5.      -> set var=parameter+1;  
  6.      -> if var=0 then 
  7.      -> insert into values(17);  
  8.      -> end if;  
  9.      -> if parameter=0 then 
  10.      -> update set s1=s1+1;  
  11.      -> else 
  12.      -> update set s1=s1+2;  
  13.      -> end if;  
  14.      -> end;  
  15.      -> //  
  16. mysql DELIMITER 
2)case語句
  1. mysql DELIMITER //  
  2. mysql CREATE PROCEDURE proc3 (in parameter int)  
  3.      -> begin 
  4.      -> declare var int;  
  5.      -> set var=parameter+1;  
  6.      -> case var  
  7.      -> when then   
  8.      -> insert into values(17);  
  9.      -> when then   
  10.      -> insert into values(18);  
  11.      -> else   
  12.      -> insert into values(19);  
  13.      -> end case;  
  14.      -> end;  
  15.      -> //  
  16. mysql DELIMITER 

4.6.2循環語句

1)while ···· end while

  1. mysql DELIMITER //  
  2. mysql CREATE PROCEDURE proc4()  
  3.      -> begin 
  4.      -> declare var int;  
  5.      -> set var=0;  
  6.      -> while var<6 do  
  7.      -> insert into values(var);  
  8.      -> set var=var+1;  
  9.      -> end while;  
  10.      -> end;  
  11.      -> //  
  12. mysql DELIMITER 

2)repeat···· end repeat:它在執行操作后檢查結果,而while則是執行前進行檢查。

  1. mysql DELIMITER //  
  2. mysql CREATE PROCEDURE proc5 ()  
  3.      -> begin   
  4.      -> declare int;  
  5.      -> set v=0;  
  6.      -> repeat  
  7.      -> insert into values(v);  
  8.      -> set v=v+1;  
  9.      -> until v>=5  
  10.      -> end repeat;  
  11.      -> end;  
  12.      -> //  
  13. mysql DELIMITER ;  

3)loop ·····end loop:loop循環不需要初始條件,這點和while 循環相似,同時和repeat循環一樣不需要結束條件, leave語句的意義是離開循環。

[begin_label:] LOOP

         statement_list
END LOOP [end_label]

  1. mysql DELIMITER //  
  2. mysql CREATE PROCEDURE proc6 ()  
  3.      -> begin 
  4.      -> declare int;  
  5.      -> set v=0;  
  6.      -> LOOP_LABLE:loop  
  7.      -> insert into values(v);  
  8.      -> set v=v+1;  
  9.      -> if >=5 then 
  10.      -> leave LOOP_LABLE;  
  11.      -> end if;  
  12.      -> end loop;  
  13.      -> end;  
  14.      -> //  
  15. mysql DELIMITER ;  

4.6.3 LEAVE語句:跳出循環控制

     LEAVE label
注:label 標號可以用在begin repeat while 或者loop 語句前,語句標號只能在合法的語句前面使用。可以跳出循環,使運行指令達到復合語句的最后一步。

4.6.4ITERATE迭代:通過引用復合語句的標號,跳出本次循環,執行下一輪

  1. mysql DELIMITER //  
  2. mysql CREATE PROCEDURE proc10 ()  
  3.      -> begin 
  4.      -> declare int;  
  5.      -> set v=0;  
  6.      -> LOOP_LABLE:loop  
  7.      -> if v=3 then   
  8.      -> set v=v+1;  
  9.      -> ITERATE LOOP_LABLE;  
  10.      -> end if;  
  11.      -> insert into values(v);  
  12.      -> set v=v+1;  
  13.      -> if v>=5 then 
  14.      -> leave LOOP_LABLE;  
  15.      -> end if;  
  16.      -> end loop;  
  17.      -> end;  
  18.      -> //  
  19. mysql DELIMITER 

4.7MySQL存儲過程的基本函數

(1).字符串類

CHARSET(str) // 返回字串字符集
CONCAT (string2 [,... ]) // 連接字串
INSTR (string ,substring ) // 返回 substring 首次在 string 中出現的位置 , 不存在返回 0
LCASE (string2 ) //
LCASE (string2 ) // 轉換成小寫
LEFT (string2 ,length ) // string2 中的左邊起取 length 個字符
LENGTH (string ) //string 長度
LOAD_FILE (file_name ) // 從文件讀取內容
LOCATE (substring , string [,start_position ] )   INSTR, 但可指定開始位置
LPAD (string2 ,length ,pad ) // 重復用 pad 加在 string 開頭 , 直到字串長度為 length
LTRIM (string2 ) //
LTRIM (string2 ) // 去除前端空格
REPEAT (string2 ,count ) // 重復 count
REPLACE (str ,search_str ,replace_str ) // str 中用 replace_str 替換 search_str
RPAD (string2 ,length ,pad) //
RPAD (string2 ,length ,pad) // str 后用 pad 補充 , 直到長度為 length
RTRIM (string2 ) //
RTRIM (string2 ) // 去除后端空格
STRCMP (string1 ,string2 ) // 逐字符比較兩字串大小 ,

SUBSTRING (str , position [,length ]) // str position 開始 , length 個字符 ,
注: mysql 中處理字符串時,默認第一個字符下標為 1 ,即參數 position 必須大於等於 1  
例:

  1. mysql> select substring('abcd',0,2);  
  2. +-----------------------+  
  3. substring('abcd',0,2) |  
  4. +-----------------------+  
  5.                       |  
  6. +-----------------------+  
  7. row in set (0.00 sec)  
  8.  
  9. mysql> select substring('abcd',1,2);  
  10. +-----------------------+  
  11. substring('abcd',1,2) |  
  12. +-----------------------+  
  13.     ab                |  
  14. +-----------------------+  
  15. row in set (0.02 sec)  
TRIM([[BOTH|LEADING|TRAILING] [padding] FROM]string2) // 去除指定位置的指定字符
UCASE (string2 ) // 轉換成大寫
RIGHT(string2,length) // string2 最后 length 個字符
SPACE(count) // 生成 count 個空格

(2).數學類

ABS (number2 ) // 絕對值
BIN (decimal_number ) // 十進制轉二進制
CEILING (number2 ) // 向上取整
CONV(number2,from_base,to_base) // 進制轉換
FLOOR (number2 ) // 向下取整
FORMAT (number,decimal_places ) // 保留小數位數
HEX (DecimalNumber ) // 轉十六進制
注: HEX() 中可傳入字符串,則返回其 ASC-11,如HEX('DEF') 返回 4142143
也可以傳入十進制整數,返回其十六進制編碼,如 HEX(25) 返回 19
LEAST (number , number2 [,..]) //
LEAST (number , number2 [,..]) // 求最小值
MOD (numerator ,denominator ) // 求余
POWER (number ,power ) // 求指數
RAND([seed]) // 隨機數

ROUND (number [,decimals ]) // 四舍五入 ,decimals 為小數位數 ]
注:返回類型並非均為整數,如:

(1)默認變為整形值

  1. mysql> select round(1.23);  
  2. +-------------+  
  3. round(1.23) |  
  4. +-------------+  
  5.           |  
  6. +-------------+  
  7. row in set (0.00 sec)  
  8.  
  9. mysql> select round(1.56);  
  10. +-------------+  
  11. round(1.56) |  
  12. +-------------+  
  13.           |  
  14. +-------------+  
  15. row in set (0.00 sec) 

(2)可以設定小數位數,返回浮點型數據

  1. mysql> select round(1.567,2);  
  2. +----------------+  
  3. round(1.567,2) |  
  4. +----------------+  
  5.           1.57 |  
  6. +----------------+  
  7. row in set (0.00 sec) 

(3).日期時間類

ADDTIME (date2 ,time_interval ) // time_interval 加到 date2
CONVERT_TZ (datetime2 ,fromTZ ,toTZ ) //
轉換時區
CURRENT_DATE ( ) //
當前日期
CURRENT_TIME ( ) //
當前時間
CURRENT_TIMESTAMP ( ) //
當前時間戳
DATE (datetime ) //
返回 datetime 的日期部分
DATE_ADD (date2 , INTERVAL d_value d_type ) //
date2 中加上日期或時間
DATE_FORMAT (datetime ,FormatCodes ) //
使用 formatcodes 格式顯示 datetime
DATE_SUB (date2 , INTERVAL d_value d_type ) //
date2 上減去一個時間
DATEDIFF (date1 ,date2 ) //
兩個日期差
DAY (date ) //
返回日期的天
DAYNAME (date ) //
英文星期
DAYOFWEEK (date ) //
星期 (1-7) ,1 為星期天
DAYOFYEAR (date ) //
一年中的第幾天
EXTRACT (interval_name FROM date ) //
date 中提取日期的指定部分
MAKEDATE (year ,day ) //
給出年及年中的第幾天 , 生成日期串
MAKETIME (hour ,minute ,second ) //
生成時間串
MONTHNAME (date ) //
英文月份名
NOW ( ) //
當前時間
SEC_TO_TIME (seconds ) //
秒數轉成時間
STR_TO_DATE (string ,format ) //
字串轉成時間 , format 格式顯示
TIMEDIFF (datetime1 ,datetime2 ) //
兩個時間差
TIME_TO_SEC (time ) //
時間轉秒數 ]
WEEK (date_time [,start_of_week ]) //
第幾周
YEAR (datetime ) //
年份
DAYOFMONTH(datetime) //
月的第幾天
HOUR(datetime) //
小時
LAST_DAY(date) //date
的月的最后日期
MICROSECOND(datetime) //
微秒
MONTH(datetime) //

MINUTE(datetime) //
返回符號 , 正負或 0
SQRT(number2) //
開平方

鏈接:







免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM