1. 程式人生 > >MySQL數據庫高級(四)——存儲過程

MySQL數據庫高級(四)——存儲過程

MySQL 存儲過程

MySQL數據庫高級(四)——存儲過程

一、存儲過程簡介

1、存儲過程簡介

存儲過程是一組具有特定功能的SQL語句集組成的可編程的函數,經編譯創建並保存在數據庫中,用戶可通過指定存儲過程的名字並給定參數來調用執行。
存儲過程是數據庫管理中常用的技術之一,可以很方便的做些類似數據統計、數據分析等工作,SQL SERVER、ORACLE、MySQL都支持存儲過程,但不同的數據庫環境語法結構有所區別。

2、存儲過程的優點

A、存儲過程增強了SQL語言的功能和靈活性。存儲過程可以用流控制語句編寫,有很強的靈活性,可以完成復雜的判斷和較復雜的運算。
B、存儲過程允許標準組件式編程。存儲過程被創建後,可以在程序中被多次調用,而不必重新編寫該存儲過程的SQL語句。而且數據庫專業人員可以隨時對存儲過程進行修改,對應用程序源代碼毫無影響。

C、存儲過程能實現較快的執行速度。如果某一操作包含大量的Transaction-SQL代碼或分別被多次執行,那麽存儲過程要比批處理的執行速度快很多。因為存儲過程是預編譯的。在首次運行一個存儲過程時查詢,優化器對其進行分析優化,並且給出最終被存儲在系統表中的執行計劃。而批處理的Transaction-SQL語句在每次運行時都要進行編譯和優化,速度相對要慢一些。
D、存儲過程能過減少網絡流量。針對同一個數據庫對象的操作(如查詢、修改),如果操作所涉及的Transaction-SQL語句被組織程存儲過程,那麽當在客戶計算機上調用該存儲過程時,網絡中傳送的只是該調用語句,從而大大增加了網絡流量並降低了網絡負載。
E、存儲過程可被作為一種安全機制來充分利用。系統管理員通過執行某一存儲過程的權限進行限制,能夠實現對相應的數據的訪問權限的限制,避免了非授權用戶對數據的訪問,保證了數據的安全。

二、存儲過程的使用

1、存儲過程的創建

創建存儲過程的語法:

CREATE
    [DEFINER = { user | CURRENT_USER }]
 PROCEDURE sp_name ([proc_parameter[,...]])
    [characteristic ...] routine_body

proc_parameter:
    [ IN | OUT | INOUT ] param_name type

characteristic:
    COMMENT ‘string‘
  | LANGUAGE SQL
  | [NOT] DETERMINISTIC
  | { CONTAINS SQL | NO SQL | READS SQL DATA | MODIFIES SQL DATA }
  | SQL SECURITY { DEFINER | INVOKER }

routine_body:
  Valid SQL routine statement

[begin_label:] BEGIN
  [statement_list]
    
END [end_label]

IN輸入參數:表示該參數的值必須在調用存儲過程時指定,在存儲過程中修改該參數的值不能被返回,為默認值。
OUT輸出參數:該值可在存儲過程內部被改變,並可返回。
INOUT輸入輸出參數:調用時指定,並且可被改變和返回。
A、無參數的存儲過程創建
創建查找平均分最高的前三名同學的存儲過程

create procedure getMax()
BEGIN
select a.sname as ‘姓名‘, AVG(b.mark) as ‘平均分‘ from 
TStudent a join TScore b on a.studentID=b.studentID
group by b.studentID order by ‘平均分‘ DESC limit 3;
END;

B、帶輸入參數的存儲過程創建
查找指定班級的平均分最高的前三名學生

create procedure getMaxByClass(in classname VARCHAR(10))
BEGIN
select a.sname as ‘姓名‘, AVG(b.mark) as ‘平均分‘ from 
TStudent a join TScore b on a.studentID=b.studentID where a.class=classname
group by b.studentID order by ‘平均分‘ DESC limit 3;
END

C、帶輸入參數和輸出參數的存儲過程創建
根據輸入的班級,找到學號最大的學生,將學號存儲到輸出參數。

create procedure getMaxSIDByClass(IN classname VARCHAR(20), out maxid int)
BEGIN
select MAX(studentID) into maxid from TStudent where class=classname;
END;

2、存儲過程的刪除

drop procedure sp_name;
不能在一個存儲過程中刪除另一個存儲過程,只能調用另一個存儲過程。

3、存儲過程的調用

call sp_name[(傳參)];
存儲過程名稱後面必須加括號,即使存儲過程沒有參數傳遞。

4、存儲過程信息的查看

show procedure status;
顯示數據庫中所有存儲的存儲過程基本信息,包括所屬數據庫,存儲過程名稱,創建時間等。
show create procedure sp_name;
顯示某一個存儲過程的詳細信息。

5、使用存儲過程插入數據

create procedure insertTStudent(in sid CHAR(5), name CHAR(10), ssex CHAR(1))
BEGIN
insert into TStudent (studentID, sname, sex)VALUES(sid, name, ssex);
select * from TStudent where studentID=sid;
END;
call insertTStudent(‘01020‘,‘孫悟空‘,‘男‘);

6、使用存儲過程刪除數據

根據提供的學號刪除先刪除學生的學生成績,再刪除學生。

create procedure deleteStudent(in sid CHAR(5))
BEGIN
delete from TScore where studentID=sid;
delete from TStudent where studentID=sid;
END;

7、使用存儲過程備份還原數據

A、使用存儲過程備份數據
創建存儲過程備份學生表,根據指定的表名創建新表,將TStudent表中的記錄導入到新表。

create procedure backupStudent(in tablename CHAR(10))
BEGIN
set @sql1=CONCAT(‘create table ‘,tablename,‘
(
studentID VARCHAR(5),
sname VARCHAR(10),
sex CHAR(1),
cardID VARCHAR(20),
Birthday DATETIME,
email VARCHAR(20),
class VARCHAR(10),
enterTime DATETIME
)‘);
prepare CT1 from @sql1;
EXECUTE CT1;

set @sql2=CONCAT(‘insert into ‘, tablename, 
‘(studentID,sname,sex,cardID,Birthday,email,class,enterTime)
select studentID,sname,sex,cardID,Birthday,email,class,enterTime from TStudent‘);
PREPARE CT2 from @sql2;
EXECUTE CT2;
END;

call backupStudent(‘table2019‘);

B、使用當前時間作為表名備份數據
創建存儲過程,使用系統當前事件構造新的表名,備份Tstudent表中的記錄。

create procedure backupStudentByDateTime()
BEGIN
DECLARE tablename VARCHAR(20);
set tablename = CONCAT(‘Table‘, REPLACE(REPLACE(REPLACE(now(),‘ ‘,‘‘),‘:‘,‘‘),‘-‘,‘‘));
set @sql1=CONCAT(‘create table ‘,tablename,‘
(
studentID VARCHAR(5),
sname VARCHAR(10),
sex CHAR(1),
cardID VARCHAR(20),
Birthday DATETIME,
email VARCHAR(20),
class VARCHAR(10),
enterTime DATETIME
)‘);
prepare CT1 from @sql1;
EXECUTE CT1;

set @sql2=CONCAT(‘insert into ‘, tablename, 
‘(studentID,sname,sex,cardID,Birthday,email,class,enterTime)
select studentID,sname,sex,cardID,Birthday,email,class,enterTime from TStudent‘);
PREPARE CT2 from @sql2;
EXECUTE CT2;
END

call backupStudentByDateTime();

C、使用存儲過程還原數據
創建存儲過程,根據輸入的學號從指定的表還原學記錄,存儲過程先刪除指定的學號的TStudent表中學生記錄,再從指定的表中插入該學生到Tstudent表。

create procedure restoreStudent(in sid VARCHAR(5), in tablename VARCHAR(20))
BEGIN
set @sql1=concat(‘delete from TStudent where studentid=‘,sid);
prepare CT1 from @sql1;
EXECUTE CT1;
set @sql2=concat(‘insert into TStudent
 (Studentid,sname,sex,cardID,Birthday,Email,Class,enterTime)
  select Studentid,sname,sex,cardID,Birthday,Email,Class,enterTime
  from ‘,tablename,‘ where studentid=‘,sid);
prepare CT2 from @sql2;
EXECUTE CT2;
END;

修改某個學生的記錄
update TStudent set sname=‘孫悟空‘ where studentID=‘00997‘;
從指定表中恢復數據
call restoreStudent(‘00997‘, ‘Table20180404215950‘);
查看恢復的結果
select * from TStudent where studentID=‘00997‘;

三、存儲過程實例

1、增加學生到數據庫表

create procedure addStudent(in num int)
begin
declare i int;
set i=1;
delete from TStudent;
while num>=i do
insert TStudent values 
(
       LPAD(convert(i,char(5)),5,‘0‘),
       CreateName(),
       if(ceil(rand()*10)%2=0,‘男‘,‘女‘),
       RPAD(convert(ceil(rand()*1000000000000000000),char(18)),18,‘0‘),
       Concat(convert(ceil(rand()*10)+1980,char(4)),‘-‘,LPAD(convert(ceil(rand()*12),char(2)),2,‘0‘),‘-‘,LPAD(convert(ceil(rand()*28),char(2)),2,‘0‘)),
       Concat(PINYIN(sname),‘@hotmail.com‘),
       case ceil(rand()*3) when 1 then ‘網絡與網站開發‘ when 2 then ‘JAVA‘ ELSE ‘NET‘ END,
       NOW()
);
set i=i+1;
end while;
select * from TStudent;
end

2、給學生添加成績

create procedure fillScore()
begin
DECLARE St_Num INT;
DECLARE Sb_Num INT;
DECLARE i1 INT;
DECLARE i2 INT;
set i1=1;
set i2=1;
delete from TScore;
select count(*) into St_Num from TStudent;
select count(*) into Sb_Num from TSubject;
while St_Num>=i1 do
set i2=1;
while Sb_Num>=i2 do
insert TScore values
 (LPAD(convert(i1,char(5)),5,‘0‘),LPAD(convert(i2,char(4)),4,‘0‘),ceil(50+rand()*50));
set i2=i2+1;
END WHILE;
set i1=i1+1;
END WHILE;
end

MySQL數據庫高級(四)——存儲過程