简而言之:存储过程是程序和sql都可以调用的,触发器不能够调用
存储过程
存储过程是一组为完成特定功能的SQL语句集,经编译后存储在数据库中,用户通过制定存储过程 的名字并给出参数来执行它,通常状况下将功能很复杂的语句编写为存储过程
- 定义 :输入参数,默认值
CREATE PROCEDURE prt(@name CHAR(20),@age INT =15 )
AS
BEGIN
PRINT @name
PRINT @age
END
- 执行存储过程
EXECUTE prt 'D',2 --指定值
EXECUTE prt 'd' --省略,使用默认值
EXECUTE prt 'x',DEFAULT --指定使用默认值
触发器基本原理:
触发器是一种实施复杂的完整性约束的特殊存储过程,在sql server进行某个特定表的修改时,右sql server自动执行。触发器一般用于加强数据库的某些使用规则。当普通的约束不足以加强数据的完整性时,就可以考虑使用触发器。
触发器触发时:
系统自动在内存中创建deleted表或inserted表;
只读,不允许修改,触发器执行完成后,自动删除。
inserted表:
临时保存了插入或更新后的记录行;
可以从inserted表中检查插入的数据是否满足业务需求;
如果不满足,则向用户发送报告错误消息,并回滚插入操作。
deleted表:
临时保存了删除或更新前的记录行;
可以从deleted表中检查被删除的数据是否满足业务需求;
如果不满足,则向用户报告错误消息,并回滚插入操作
- 创建
create trigger trigger_name
on {table_name view_name}
{for After Instead of }
[ insert, update,delete ]
as
sql_statement
- 修改
alter trigger trigger_name
on {table_name view_name}
{for After Instead of }
[ insert, update,delete ]
as
sql_statement
- 删除
drop trigger trigger_name
一些啰嗦的概念开始
存储过程即为能完成特定功能的一组SQL语句集。如果需要对查出的多条数据进行操作的话,这里需要理解游标(CURSOR)的概念,对于oracle有for each row命令,可以不用游标。
游标可以理解为一个结果集,描述cursor的关键字有:
SCROLL关键字指明游标可以在任意方向上滚动
FORWARD_ONLY指明游标只能向前滚动。
READ ONLY指明在游标结果集中不允许进行数据修改。
UPDATE关键字指明游标的结果集可以修改。
GLOBAL关键字使得游标对于整个连接全局可见
LOCAL关键字指明游标是局部的,它只能在它所声明的过程中使用。
等等。。。
一般结果集需要遍历,如例,将一些表中的字段查出来,插入到另外一张表中:
例1:查询表数据插入到另外一张表
CREATE PROCEDURE sp_insertIntoTest_hibernate
AS
DECLARE @projectid VARCHAR(20)
DECLARE @projectcode VARCHAR(20)
DECLARE @projectname VARCHAR(100)
DECLARE @projectamount DECIMAL(16,2)
DECLARE @remark VARCHAR(1000)
DECLARE cs CURSOR FOR --定义游标的数据集
SELECT L.PROJECTID,P.CODE,L.PROJECTNAME,L.LEASINGAMOUNT, L.proInfoSummary
FROM LAPROJECT L LEFT JOIN PARTY_CUSTOMER P ON P.ID=L.TENANTID
WHERE (L.TENANTID<>'' AND L.TENANTID IS NOT NULL) ORDER BY L.PROJECTID
BEGIN
DELETE FROM test_hibernate
OPEN cs --打开游标
FETCH NEXT FROM cs INTO @projectid,@projectcode,@projectname,@projectamount,@remark --从游标中取值赋给变量
WHILE @@FETCH_STATUS = 0 --fetch语句执行成功返回0,fetch语句执行失败或者此行不在结果集中返回-1,被提取的行不存在则返回-2.
BEGIN
PRINT '====start insert===='
INSERT INTO test_hibernate VALUES(@projectid,@projectcode,@projectname,@projectamount,@remark)
PRINT '==== end insert===='
FETCH NEXT FROM cs INTO @projectid,@projectcode,@projectname,@projectamount,@remark --继续取下一行数据
END
CLOSE cs --关闭游标
DEALLOCATE cs --删除游标
END
一些啰嗦的概念结束
实用案例
- 创建表
create table zhuangzhan
(
name nvarchar(10),
code varchar(20)
);
- 往表添加一列
alter table zhuangzhan add descition int;
- 添加数据
insert zhuangzhan values('王五','1000003',10000);
insert zhuangzhan values('李四','1000002',0);
- 查询验证
select * from zhuangzhan;
- 给表创建触发器用于添加数据出发,不允许添加code相同的数据
算啦,不推荐使用触发器,管理系统可以使用,不过特别注意锁表的问题,用多了容易锁死