存储过程的概念
存储过程(Stored Procedure)是一组为了完成特定功能的SQL 语句集,经编译后存储在数据库。中用户通过指定存储过程的名字并给出参数(如果该存储过程带有参数)来执行它。
在SQL Server 的系列版本中存储过程分为两类:系统提供的存储过程和用户自定义存储过程。系统过程主要存储在master 数据库中并以sp_为前缀,并且系统存储过程主要是从系统表中获取信息,从而为系统管理员管理SQL Server 提供支持。通过系统存储过程,MS SQL Server 中的许多管理性或信息性的活动(如了解数据库对象、数据库信息)都可以被顺利有效地完成。尽管这些系统存储过程被放在master 数据库中,但是仍可以在其它数据库中对其进行调用,在调用时不必在存储过程名前加上数据库名。而且当创建一个新数据库时,一些系统存储过程会在新数据库中被自动创建。用户自定义存储过程是由用户创建并能完成某一特定功能(如查询用户所需数据信息)的存储过程。
1) 存储过程允许标准组件式编程
存储过程在被创建以后可以在程序中被多次调用,而不必重新编写该存储过程的SQL 语句。而且数据库专业人员可随时对存储过程进行修改,但对应用程序源代码毫无影响(因为应用程序源代码只包含存储过程的调用语句),从而极大地提高了程序的可移植性。
(2) 存储过程能够实现较快的执行速度
如果某一操作包含大量的Transaction-SQL 代码或分别被多次执行,那么存储过程要比批处理的执行速度快很多。因为存储过程是预编译的,在首次运行一个存储过程时,查询优化器对其进行分析、优化,并给出最终被存在系统表中的执行计划。而批处理的Transaction- SQL 语句在每次运行时都要进行编译和优化,因此速度相对要慢一些。
(3) 存储过程能够减少网络流量
对于同一个针对数据数据库对象的操作(如查询、修改),如果这一操作所涉及到的 Transaction-SQL 语句被组织成一存储过程,那么当在客户计算机上调用该存储过程时,网络中传送的只是该调用语句,否则将是多条SQL 语句,从而大大增加了网络流量,降低网络负载。
(4) 存储过程可被作为一种安全机制来充分利用
系统管理员通过对执行某一存储过程的权限进行限制,从而能够实现对相应的数据访问权限的限制,避免非授权用户对数据的访问,保证数据的安全。(我们将在14 章“SQLServer 的用户和安全性管理”中对存储过程的这一应用作更为清晰的介绍)
注意:存储过程虽然既有参数又有返回值,但是它与函数不同。存储过程的返回值只是指明执行是否成功,并且它不能像函数那样被直接调用,也就是在调用存储过程时,在存储过程名字前一定要有EXEC保留字。
示例代码:
实验代码如下:
use student
/*1.创建一个存储过程my_proc,查询“学生表”中所有计算机系女生的学号、姓名、性别、年龄和所在院系。*/
create procedure my_proc
as
select 学号,姓名,性别,年龄,所在院系 from 学生表 where 性别='女'and 所在院系='计算机'
return
/*(2)执行存储过程*/
exec my_proc
/*3.修改存储过程,使其能够查询计算机系女生的所有基本信息*/
alter procedure my_proc
as
select * from 学生表 where 性别='女'and 所在院系='计算机'
return
exec my_proc
drop procedure my_proc
/*1 .带输入参数的存储过程
(1)创建一个存储过程my_procsex,使其能够查询“学生表”中男学生或女学生的学号、姓名、性别、年龄和所在院系
(2)执行存储过程*/
USE STUDENT
create procedure my_procsex
@sex char(2),
@sex2 char(2)
as
select 学号,姓名,性别,年龄,所在院系 from 学生表
where 性别=@sex or 性别=@sex2
return
drop procedure my_procsex
exec my_procsex @sex='男',@sex2='女'
/*创建一个存储过程my_procage,使其能够根据学生姓名,查询学生年龄。
(考虑当学生不存在时给出提示信息)*/
create procedure my_procage
@name char(20)
as
if not exists(
select 年龄 from 学生表 where 姓名=@name)
print '没有找到该学生信息'
return
exec my_procage @name='a'
drop procedure my_procage
/*1)创建一个存储过程my_procstatus,使其能够根据学生姓名,查询学生的选课信息。
(如果没有输入学生姓名,返回状态码55;如果输入的学生姓名不存在,则返回状态码-155)*/
drop procedure my_procstatus
create procedure my_procstatus
@name varchar(10)
as
if @name=''
return 55
select 成绩表.学号,课程号 from 成绩表,学生表 where
成绩表.学号=学生表.学号 and 学生表.姓名=@name
if not exists(
select 成绩表.学号,课程号 from 成绩表,学生表 where
成绩表.学号=学生表.学号 and 学生表.姓名=@name
)
return -155
select 成绩表.学号,课程号 from 成绩表,学生表 where
成绩表.学号=学生表.学号 and 学生表.姓名=@name
return
declare @status int
exec @status=my_procstatus @name=''
if @status=-155
print'没有找到'
if @status=55
print'没有输入名字'
/*在存储过程中使用临时表,不指定题目,请理解全局临时表与局部临时表的区别*/
create procedure my_procsave
as
select 姓名 from (select * from 学生表 where 所在院系='计算机')#A
where #A.性别='女'
return
exec my_procsave
drop procedure my_procsave
create procedure my_procsave
as
select 姓名 from (select * from 学生表 where 所在院系='计算机')##A
where ##A.性别='女'
return
exec my_procsave