一、觸發器
觸發器(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 觸發器名 ;
二、存儲過程和函數
存儲過程(Stored Procedure),是一組為了完成特定功能的SQL 語句,集經編譯后存儲在數據庫中,用戶通過指定存儲過程的名字並給出參數,如果該存儲過程帶有參數來執行。
1、MySQL存儲過程的創建
1.1格式
CREATE PROCEDURE 過程名 ([過程參數[,...]])
[特性 ...] 過程體
1.2例:
- mysql>
DELIMITER // - mysql>
CREATE PROCEDURE proc1( OUTs int) -
BEGIN-> -
SELECT-> COUNT(*) INTO s FROMuser; -
END-> -
-> // - 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參數例子
- mysql
> DELIMITER // - mysql
> CREATEPROCEDURE demo_in_parameter( INp_in int) - ->
BEGIN - ->
SELECT p_in; - ->
SET p_in=2; - ->
SELECT p_in; - ->
END; - ->
// - mysql
> DELIMITER ;
執行結果:
- mysql
> SET@p_in=1; - mysql
> CALL demo_in_parameter(@p_in); - +------+
- |
p_in | - +------+
- |
1 | - +------+
-
- +------+
- |
p_in | - +------+
- |
2 | - +------+
-
- mysql>
SELECT @p_in; - +-------+
- |
@p_in | - +-------+
- |
1 | - +-------+
以上可以看出,p_in雖然在存儲過程中被修改,但並不影響@p_id的值
- mysql
> DELIMITER // - mysql
> CREATEPROCEDURE demo_out_parameter( OUTp_out int) - ->
BEGIN - ->
SELECT p_out; - ->
SET p_out=2; - ->
SELECT p_out; - ->
END; - ->
// - mysql
> DELIMITER ;
執行結果:
- mysql
> SET@p_out=1; - mysql
> CALL sp_demo_out_parameter(@p_out); - +-------+
- |
p_out | - +-------+
- |
NULL | - +-------+
-
- +-------+
- |
p_out | - +-------+
- |
2 | - +-------+
-
- mysql>
SELECT @p_out; - +-------+
- |
p_out | - +-------+
- |
2 | - +-------+
3)
- mysql
> DELIMITER // - mysql
> CREATEPROCEDURE demo_inout_parameter(INOUT int)p_inout - ->
BEGIN - ->
SELECT p_inout; - ->
SET p_inout=2; - ->
SELECT p_inout; - ->
END; - ->
// - mysql
> DELIMITER ;
- mysql
> SET p_inout=1;@ - mysql
> CALL demo_inout_parameter(@p_inout) ; - +---------+
- |
p_inout | - +---------+
- |
1 | - +---------+
-
- +---------+
- |
p_inout | - +---------+
- |
2 | - +---------+
-
- mysql
> SELECT @p_inout; - +----------+
- |
@p_inout | - +----------+
- |
2 | - +----------+
1.4變量
1)定義變量
DECLARE variable_name [,variable_name...] datatype [DEFAULT value];
注:datatype為MySQL的數據類型,如:int, float, date, varchar(length)
例如:
- DECLARE
l_int intunsigned default4000000; - DECLARE
l_numeric DEFAULTnumber(8,2) 9.95; - DECLARE
l_date dateDEFAULT '1999-12-31'; - DECLARE
l_datetime DEFAULTdatetime '1999-12-31 23:59:59' ; - DECLARE
l_varchar varchar(255)DEFAULT 'This will ;not be padded'
2)變量賦值
SET 變量名 = 表達式值 [,variable_name = expression ...]
3)用戶變量
- 在MySQL客戶端使用用戶變量
- mysql
> SELECT'Hello World' into @x; - mysql
> SELECT@x; - +-------------+
- |
@x | - +-------------+
- |
Hello World | - +-------------+
- mysql
> SET@y= 'GoodbyeCruel ;World' - mysql
> SELECT@y; - +---------------------+
- |
@y | - +---------------------+
- |
Goodbye Cruel World | - +---------------------+
-
- mysql
> SET@z=1+2+3; - mysql
> SELECT@z; - +------+
- |
@z | - +------+
- |
6 | - +------+
- 在存儲過程中使用用戶變量
- mysql
> CREATEPROCEDURE GreetWorld( SELECT) CONCAT(@greeting, 'World' ); - mysql
> SET@greeting= 'Hello'; - mysql
> CALL GreetWorld( ); - +----------------------------+
- |
CONCAT(@greeting, 'World' )| - +----------------------------+
- |
Hello World | - +----------------------------+
- 在存儲過程間傳遞全局范圍的用戶變量
- mysql>
CREATE PROCEDURE p1() SET@last_procedure= 'p1'; - mysql>
CREATE PROCEDURE p2() SELECTCONCAT( 'Lastprocedure ,@last_proc);was ' - mysql>
CALL p1( ); - mysql>
CALL p2( ); - +-----------------------------------------------+
- |
CONCAT( 'Lastprocedure ,@last_procwas ' | - +-----------------------------------------------+
- |
Last procedure was p1 | - +-----------------------------------------------+
注:
1)
用戶變量名一般以@開頭,
濫用用戶變量會導致程序難以理解及管理。
2)
CONCAT(str1,str2,…) 連接函數
返回結果為連接參數產生的字符串。如有任何一個參數為NULL ,則返回值為 NULL。
例:
mysql> select concat('11','22','33');
+------------------------+
| concat('11','22','33') |
+------------------------+
| 112233 |
+------------------------+
+------------------------+
| concat('11','22','33') |
+------------------------+
| 112233 |
+------------------------+
4)變量作用域
內部的變量在其作用域范圍內享有更高的優先權,當執行到end。變量時,內部變量消失,此時已經在其作用域外,變量不再可見了,此時在存儲
過程外再也不能找到這個申明的變量,但是你可以通過out參數或者將其值指派給會話變量來保存其值。
例:
- mysql
> DELIMITER // - mysql
> CREATEPROCEDURE proc3() -
begin-> -
declare-> x1 varchar(5)default 'outer'; -
begin-> -
declare-> x1 varchar(5)default 'inner'; -
select-> x1; -
end;-> -
select-> x1; -
end;-> -
-> // - mysql
> DELIMITER ;
1.5注釋
MySQL存儲過程可使用兩種風格的注釋
雙模杠(--):
該風格一般用於單行注釋
?c風格:一般用於多行注釋
例:
mysql
- mysql
> CREATEPROCEDURE proc1 --name存儲過程名 - ->
( INparameter1 INTEGER) - ->
BEGIN - ->
DECLARE variable1 CHAR(10); - ->
IF THENparameter1 = 17 - ->
SET variable1 'birds';= - ->
ELSE - ->
SET variable1 'beasts';= - ->
END IF; - ->
INSERT INTO table1 VALUES(variable1); - ->
END - ->
// - 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語句
- mysql
> DELIMITER // - mysql
> CREATEPROCEDURE proc2( INparameter int) -
begin-> -
declare-> var int; -
set-> var=parameter+1; -
then-> if var=0 -
insert-> into t values(17); -
end-> if ; -
then-> if parameter=0 -
update-> t sets1=s1+1; -
else-> -
update-> t sets1=s1+2; -
end-> if ; -
end;-> -
-> // - mysql
> DELIMITER
2)case語句
- mysql
> DELIMITER // - mysql
> CREATEPROCEDURE proc3 in( parameter int) -
begin-> -
declare-> var int; -
set-> var=parameter+1; -
case-> var -
when-> 0 then -
insert-> into t values(17); -
when-> 1 then -
insert-> into t values(18); -
else-> -
insert-> into t values(19); -
end-> case; -
end;-> -
-> // - mysql
> DELIMITER ;
4.6.2循環語句
1)while ···· end while
- mysql
> DELIMITER // - mysql
> CREATEPROCEDURE proc4() -
begin-> -
declare-> var int; -
set-> var=0; -
-> while var<6 do -
insert-> into t values(var); -
set-> var=var+1; -
end-> while ; -
end;-> -
-> // - mysql
> DELIMITER ;
2)repeat···· end repeat:它在執行操作后檢查結果,而while則是執行前進行檢查。
- mysql
> DELIMITER // - mysql
> CREATEPROCEDURE proc5 () -
begin-> -
declare-> v int; -
set-> v=0; -
-> repeat -
insert-> into t values(v); -
set-> v=v+1; -
-> until v>=5 -
end-> repeat ; -
end;-> -
-> // - mysql
> DELIMITER ;
3)loop ·····end loop:loop循環不需要初始條件,這點和while 循環相似,同時和repeat循環一樣不需要結束條件, leave語句的意義是離開循環。
[begin_label:] LOOP
statement_list
END LOOP [end_label]
- mysql
> DELIMITER // - mysql
> CREATEPROCEDURE proc6 () -
begin-> -
declare-> v int; -
set-> v=0; -
-> LOOP_LABLE:loop -
insert-> into t values(v); -
set-> v=v+1; -
then-> if v >=5 -
-> leave LOOP_LABLE; -
end-> if ; -
end-> loop ; -
end;-> -
-> // - mysql
> DELIMITER ;
4.6.3 LEAVE語句:跳出循環控制
LEAVE label
注:label
標號可以用在begin repeat while 或者loop 語句前,語句標號只能在合法的語句前面使用。可以跳出循環,使運行指令達到復合語句的最后一步。
4.6.4ITERATE迭代:通過引用復合語句的標號,跳出本次循環,執行下一輪。
- mysql
> DELIMITER // - mysql
> CREATEPROCEDURE proc10 () -
begin-> -
declare-> v int; -
set-> v=0; -
-> LOOP_LABLE:loop -
then-> if v=3 -
set-> v=v+1; -
-> ITERATE LOOP_LABLE; -
end-> if; -
insert-> into t values(v); -
set-> v=v+1; -
then-> if v>=5 -
-> leave LOOP_LABLE; -
end-> if; -
end-> loop; -
end;-> -
-> // - 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
例:
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
例:
- mysql>
select substring('abcd',0,2); - +-----------------------+
- |
substring('abcd',0,2) | - +-----------------------+
- |
| - +-----------------------+
- 1
row inset (0.00 sec) -
- mysql>
select substring('abcd',1,2); - +-----------------------+
- |
substring('abcd',1,2) | - +-----------------------+
- |
ab | - +-----------------------+
- 1
row inset (0.02 sec)
TRIM([[BOTH|LEADING|TRAILING] [padding] FROM]string2) //
去除指定位置的指定字符
UCASE (string2 ) // 轉換成大寫
RIGHT(string2,length) // 取 string2 最后 length 個字符
SPACE(count) // 生成 count 個空格
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]) // 隨機數
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)默認變為整形值
- mysql>
select round(1.23); - +-------------+
- |
round(1.23) | - +-------------+
- |
1 | - +-------------+
- 1
row inset (0.00 sec) -
- mysql>
select round(1.56); - +-------------+
- |
round(1.56) | - +-------------+
- |
2 | - +-------------+
- 1
row inset (0.00 sec)
(2)可以設定小數位數,返回浮點型數據
- mysql>
select round(1.567,2); - +----------------+
- |
round(1.567,2) | - +----------------+
- |
1.57 | - +----------------+
- 1
row inset (0.00 sec)
(3).日期時間類
ADDTIME (date2 ,time_interval ) // 將 time_interval 加到 date2CONVERT_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) // 開平方
鏈接:
