上一级目录:Mosh_完全掌握SQL课程_学习笔记
其它相关:数据概要
【第九章】存储过程
Stored Procedures (时长48分钟)
1. 什么是存储过程
What are Stored Procedures (2:18)
小结
存储过程三大作用:
- 储存和管理SQL代码 Store and organize SQL
- 性能优化 Faster execution
- 数据安全 Data security
导航
之前学了增删改查,包括复杂查询以及如何运用视图来简化查询。
假设你要开发一个使用数据库的应用程序,你应该将SQL语句写在哪里呢?
如果将SQL语句内嵌在应用程序的代码里,将使其混乱且难以维护,所以应该将SQL代码和应用程序代码分开,将SQL代码储存在所属的数据库中,具体来说,是放在储存过程(stored procedure)和函数中。
储存过程是一个包含SQL代码模块的数据库对象,在应用程序代码中,我们调用储存过程来获取和保存数据(get and save the data)。也就是说,我们使用储存过程来储存和管理SQL代码。
使用储存程序还有另外两个好处。首先,大部分DBMS会对储存过程中的代码进行一些优化,因此有时储存过中的SQL代码执行起来会更快。
此外,就像视图一样,储存过程能加强数据安全。比如,我们可以移除对所有原始表的访问权限,让各种增删改的操作都通过储存过程来完成,然后就可以决定谁可以执行何种储存过程,用以限制用户对我们数据的操作范围,例如,防止特定的用户删除数据。
所以,储存过程很有用,本章将学习如何创建和使用它。
2. 创建一个存储过程
Creating a Stored Procedure (5:34)
小结
DELIMITER $$
-- delimiter n. 分隔符
CREATE PROCEDURE 过程名()
BEGIN
……;
……;
……;
END$$
DELIMITER ;
实例
创造一个get_clients()过程
CREATE PROCEDURE get_clients()
-- 括号内可传入参数,之后会讲
-- 过程名用小写单词和下划线表示,这是约定熟成的做法
BEGIN
SELECT * FROM clients;
END
BEGIN 和 END 之间包裹的是此过程(PROCEDURE)的内容(body),内容里可以有多个语句,但每个语句都要以 ;
结束,包括最后一个。
为了将过程内容内部的语句分隔符与SQL本身执行层面的语句分隔符 ;
区别开,要先用 DELIMITER(分隔符) 关键字暂时将SQL语句的默认分隔符改为其他符号,一般是改成双美元符号 $$
,创建过程结束后再改回来。注意创建过程本身也是一个完整SQL语句,所以别忘了在END后要加一个暂时语句分隔符 $$
注意
过程内容中所有语句都要以 ;
结尾并且因此要暂时修改SQL本身的默认分隔符,这些都是MySQL地特性,在SQL Server等就不需要这样
DELIMITER $$
CREATE PROCEDURE get_clients()
BEGIN
SELECT * FROM clients;
END$$
DELIMITER ;
调用此程序:
法1. 点击闪电按钮
法2. 用CALL关键字
USE sql_invoicing;
CALL get_clients()
或
CALL sql_invoicing.get_clients()
注意
上面讲的是如何在SQL中调用储存过程,但更多的时候其实是要在应用程序代码(可能是 C#、JAVA 或 Python 编写的)中调用。
练习
创造一个储存过程 get_invoices_with_balance(取得有差额(差额不为0)的发票记录)
DROP PROCEDURE get_invoices_with_balance;
-- 注意DROP语句里这个过程没有带括号
DELIMITER $$
CREATE PROCEDURE get_invoices_with_balance()
BEGIN
SELECT *
FROM invoices_with_balance
-- 这是之前创造的视图
-- 用视图好些,因为有现成的balance列
WHERE balance > 0;
END$$
DELIMITER ;
CALL get_invoices_with_balance();
3. 使用MySQL工作台创建存储过程
Creating Procedures Using MySQLWorkbench (1:21)
也可以用点击的方式创造过程,右键选择 Create Stored Procedure,填空,Apply。这种方式 Workbench 会帮你处理暂时修改分隔符的问题
这种方式一样可以储存SQL文件
事实证明,mosh很喜欢用这种方式,后面基本都是用这种方式创建过程(毕竟不用管改分隔符的问题,能偷懒又何必自找麻烦呢