sql插入触发器
‘壹’ 有关sql insert触发器和update触发器
DML触发器有三类:
1, insert触发器;
2, update触发器;
3, delete触发器;
触发器的组成部分:
触发器的声明,指定触发器定时,事件,表名以类型
触发器的执行,PL/SQL块或对过程的调用
触发器的限制条件,通过where子句实现
类型:
应用程序触发器,前台开发工具提供的;
数据库触发器,定义在数据库内部由某种条件引发;分为:
DML触发器;
数据库级触发器;
替代触发器;
DML触发器组件:
1,触发器定时
2,触发器事件
3,表名
4, 触发器类型
5, When子句
6, 触发器主体
可创建触发器的对象:数据库表,数据库视图,用户模式,数据库实例
创建DML触发器:
Create [or replace] trigger [模式.]触发器名
Before| after insert|delete|(update of 列名)
On 表名
[for each row]
When 条件
PL/SQL块
For each row的意义是:在一次操作表的语句中,每操作成功一行就会触发一次;不写的话,表示是表级触发器,则无论操作多少行,都只触发一次;
When条件的出现说明了,在DML操作的时候也许一定会触发触发器,但是触发器不一定会做实际的工作,比如when 后的条件不为真的时候,触发器只是简单地跳过了PL/SQL块;
Insert触发器的创建:
create or replace trigger tg_insert
before insert on student
begin
dbms_output.put_line('insert trigger is chufa le .....');
end;
/
执行的效果:
SQL> insert into student
2 values(202,'dongqian','f');
insert trigger is chufa le .....
update表级触发器的例子:
create or replace trigger tg_updatestudent
after update on student
begin
dbms_output.put_line('update trigger is chufale .....');
end;
/
运行效果:
SQL> update student set se='f';
update trigger is chufale .....
已更新8行;
可见,表级触发器在更新了多行的情况下,只触发了一次;
如果在after update on student后加上
For each row的话就成为行级触发器,运行效果:
SQL> update student set se='m';
update trigger is chufale .....
update trigger is chufale .....
update trigger is chufale .....
update trigger is chufale .....
update trigger is chufale .....
update trigger is chufale .....
update trigger is chufale .....
update trigger is chufale .....
已更新8行;
:new 与: old:必须是针对行级触发器的,也就是说要使用这两个变量的触发器一定有for each row
这两个变量是系统自动提供的数组变量,:new用来记录新插入的值,old用来记录被删除的值;
使用insert的时候只有:new里有值;
使用delete的时候只有:old里有值;
使用update的时候:new和:old里都有值;
可以这样使用: dbms_output.put_line('insert trigger is chufa
dbms_output.put_line('new id is : '||:new.stui
dbms_output.put_line('new name is : '||:new.st
dbms_output.put_line('new se is : '||:new.se);
可以这样从数据字典中查看一个表上有哪几个触发器:
SQL> select trigger_name from user_triggers
2 where table_name=upper('student');
TRIGGER_NAME
------------------------------
TG_INSERT
TG_UPDATESTUDENT
带有:old变量的行级delete触发器:
create or replace trigger tg_deletestudent
before delete on student
for each row
begin
dbms_output.put_line('old is: '||:old.stuid);
dbms_output.put_line('old name: '||:old.stuname);
end;
/
运行效果:
SQL> delete from student;
old is: 202
old name: dongqian
old is: 101
old name: liudehua
old is: 102
old name: lingqingxia
old is: 103
old name: lichanggong
old is: 104
old name: zhenxiuwen
old is: 1001
old name: lilianjie
old is: 1009
old name: tongleifuck
old is: 203
old name: kfdj
old is: 209
old name: fuck
已删除9行
When的使用:如果在begin也就是说触发器的PL/SQL主体块执行前加上when(old.se=’f’)的话,DML操作照做不误,但是只会在删除
Se=’f’的那行的时候才会执行触发器的主体动作,执行效果:
SQL> delete from student;
old is: 209
old name: fuck
已删除9行; 这里虽然删了9行,但是只执行了一次触发器的主体,做为一个行级触发器;
混合类型触发器:
Inserting,deleting,updating三个谓词可以分别指示当前操作到底是哪个;
create or replace trigger hunhetrigger
before insert or update or delete on student
for each row
begin
if inserting then
dbms_output.put_line('insert le.........');
end if;
if deleting then
dbms_output.put_line('delete le .......');
end if;
end;
/
插入的时候就自动判断当前动作为插入:
SQL> insert into student values(303,'me','f');
insert le.........
删除的时候就自动判断当前动作为删除:
SQL> delete from student;
delete le .......
注意,既然触发器内部的主体PL/SQL是语句,那么它同样也可以是插入删除操作而不一定只是dbms_output打印一些信息;
这正是日志表的原理:在用户执行了DML语句的时候触发主体为插入日志表以记录操作轨迹的触发器;
为什么用触发器? 当我们有两个表用来记录商品的出库入库情况,good_store用来记录库存的产品类别和数量,
而good_out用来记录出库的产品类别和数量,那么每当我们出库的某个类别的产品一定数量的时候,我们应该在good_out中插入该产品的类别和
出库数量,而同时也应该在good_store表中用update来更新库存的相应类别的产品的数量;这就交给了我们两个必须完成的任务:插入good_out
表后更新good_store表,这样的手工过程使得我们觉得非常ugly,如果只做其中一个那造成数据的不一致;所以现在我们可以用触发器,在
Good_out表的插入操作上绑定一个对good_store进行更新的触发器;当然这个过程应该是一个事务,你不必担心插入good_out表执行了,而绑定在这个动作上的触发器操作不会执行,相信Oracle设计为原子性了;
注意:触发器会使得原来的SQL语句速度变慢;
替代触发器:
创建在视图上的触发器,就是替代触发器,只能是行级触发器;
为什么要用替代触发器?
假如你有一个视图是基于多个表的字段连接查询得到的;现在如果你想直接对着这个视图insert;那你一定在想,我对视图的插入操作
怎么来反应到组成这个视图的各个表中呢?事实上,除了定义一个触发器来绑定在对视图上的插入动作上外,你没有别的办法通过系统的报错而直接向视图中插入数据;这就是我们用替代触发器的原因;替换的意思实际上是触发器的主体部分把对视图的插入操作转换成详细的对各个表的插入;
变异表:变异表就是当前SQL语句正在修改的表,所以在一个变异表上绑定的触发器不可以使用cout()函数,原因很简单:SQL语句刚刚修改了表,你怎么统计??
约束表:
维护:
Alter trigger …..disenable; 使得触发器不可用;
Alter trigger ……enable; 开启触发器;
Oracle的内置程序包
扩展数据库的功能;
为PL/SQL提供对SQL功能的访问;
一般具有sys权限的高级管理人员使用;
一个典型的程序包就是dbms_output,你老是用它的过程put_line();
Dbms_standard 提供语言工具;
Dbms_lob操作Oracle LOB;就是针对大型数据的操作设计的;
Dbms_lock用户定义的锁;
Dbms_job 允许对PL/SQL过程进行调度;
Dbms_alert 支持数据库事件的异步通知;
1,dbms_output的一些过程:
a):enable
b):disable
c):put只是把数据放到缓存(SQL-Plus的缓存,实际就是整个窗口)中,无输出功能;
d):put_line可以使得以前放在缓存中所有数据输出;并且换到下一行;
e):new_line
f):get_line
g):get_lines
2,dmbs_lob ,这个包只能是由系统管理员来操作;
Clob以字符数据存储可达2G;
Blob以二进制数据存储可达4G;
Nclob以unicode字符存储;
一个文件下载列表的例子:
创建下载目录表:
create table downfilelist
(
id varchar(20) not null primary key,
name varchar(40) not null,
filelocation bfile,
description clob
)
/
创建目录:
create or replace directory filedir as 'f:\oracle'
/只是向Oralce注册了目录,实际上并不会真的建立目录在磁盘上;Oracle无权管理和锁定操作系统的文件系统;
向目录表中插入数据:
insert into downfilelist
values('10001','oracle plsal编程指南',bfilename(upper('filedir'),'demo.mp3'),'this is a mp3 music')
insert into downfilelist
values('10002','java 大权', bfilename(upper('filedir'),'x.jpg'),'good super girl')
/在filedir的目录f:\oracle下实际存储着demo.mp3 ,x.jpg;
注意,如果你试图查询,效果是 :
sys>select * from downfilelist;
SP2-0678: 列或属性类型无法通过 SQL*Plus 显示
因为第三列是无法显示的,是一个二进制的;
下面使用dbms_lob的一些过程来进行操作:
1,read过程
declare
tempdesc clob;
ireadcount int;
istart int;
soutputdesc varchar(100);
begin
ireadcount:=5;
istart:=1;
select description into tempdesc from downfilelist where id='10001';
dbms_lob.read(tempdesc,ireadcount,istart,soutputdesc); 把clob类型的tempdesc中的数据读到字符类型的soutputdesc里;
dbms_output.put_line('Top 5 character is: '||soutputdesc);
end;
/注意,对unicode来说,汉字和字母所占的位数是一样的;
2,getlength函数
select description into tempclob from downfilelist where id=‘10001’;
ilen:=dbms_lob.GetLength(tempclob);
append,……..
发现这样的现象:select x into y的时候,y并不是独立于x的拷贝,因为当修改y的时候x也被修改了;
3, fileexists函数
select id ,dbms_lob.fileexists(filelocation) from downfilelist;
如果在bfile类型字段filelocation指定的系统下的目录中存在filelocation指定的系统文件,则返回int 1,否则返回0;
这说明Oracle还是可以检测到系统的文件情况的,如同java.io包里的类一样;
对bfile类型数据的操作函数有fileisopen,fileopen,fileclose等等;
如果对您有帮助,请记得采纳为满意答案,谢谢!祝您生活愉快!
vaela
‘贰’ SQl中触发器怎样执行的
创建触发器 是特殊的存储过程,自动执行,一般不要有返回值。
1、后触发器 (AFTER,FOR)先执行对应语句,后执行触发器中的语句。
2、前触发器 并没有真正的执行触发语句(insert,update,delete),而是执行触发后的语句。
3、行级触发器 (FOR EACH ROW) 在SQL server 中不存在。
(2)sql插入触发器扩展阅读:
创建触发的语法
CREATE TRIGGER trigger_name --触发器名称
ON table_name --触发的表
[WITH ENCRYPTION]
FOR [DELETE, INSERT, UPDATE] --选择触发器类型
AS --触发后要做的语句
T-SQL语句
GO --结束标记
‘叁’ sql插入触发器使得有成绩录入时课程表中的选课人数自动加一
摘要 create or replace trigger aaa
‘肆’ sql 增加 触发器
drop TRIGGER if EXISTS insert_data;
create trigger insert_data
after INSERT on a
for each ROW
BEGIN
insert into b select * from a order by id desc limit 1;
end;
a表的id设置自增,每次插入数据的时候获取a表最新一条id,插入到B表,亲测有效
‘伍’ sql插入一条语句触发器
/* 如果存在同名的触发器,则删除之* /
IF EXISTS ( SELECT name FROM sysobjects
WHERE type = ‘ TR‘ AND name = 'student_insert' )
DROP TRIGGER student_insert
GO
/*建立新的触发器(在向表中插入数据时触发)*/
CREATE TRIGGER student_insert ON student
FOR INSERT
AS
/*声明触发器运行时要用到的变量*/
DECLARE @NumOfStudent TINYINT
/%将插入记录的那个班级的学生总数赋给上面的变量%/
SELECT @NumOfStudent = c.student_num FROM class c ,inserted i
WHERE c.class_id = i.class_id
IF (@NumOfStudent > 0)/*如果class_id这个班级的学生总数不是0个*/
BEGIN
/*将班级学生总数+1*/
UPDATE class SET student_num = student_num + 1
FROM class c ,inserted i WHERE c.class_id = i.class_id
END
ELSE /*否则,统计一下student表中,属于class_id的学生数,并赋给班级表中的student_num字段*/
BEGIN
UPDATE class SET student_num = (
SELECT COUNT(s.student_id) FROM student s ,inserted i
WHERE s.class_id = i.class_id)
FROM class c, inserted i WHERE c.class_id = i.class_id
END
GO
‘陆’ SQL insert触发器问题
你用的for insert 肯定是不行的
在你执行触发器检测之前,sql就已经插入数据到你的数据库中去了
解决方案有两种
1 是不用触发器,自己在插入数据之前手动进行检测,如果存在,则不在插入
2 把for insert改成 instead of insert
sql中的前置触发器是instead of,这种触发器在数据更新前发生
但要注意,在检查数据通过后,要在该触发器中重发更新语句,否则不会发生更新动作.
就是说
if exists(select * from Client where ClientName=@ClientName)
begin
rollback tran
raiserror('该客户名已存在,请重新输入客户名!',16,1)
end
这里还要改 改成
if exists(select * from Client where ClientName=@ClientName)
begin
rollback tran
raiserror('该客户名已存在,请重新输入客户名!',16,1)
end
else
insert into Client ...(你的插入语句)
总的说来 第一方法简单些,而且性能相对来说好一些
‘柒’ 求SQL插入触发器语句
这个其实不是很难,其sql语句为;
create trigger a_trigger
on TB2
for insert
as
begin
update TB1 set b=b+1 from inserted,TB1 where inserted.a=TB1.a
end
经过在sql 2005/sql 2008完美通过测试!
‘捌’ SQL Server Insert触发器
年龄+1岁是指他们本来有年龄数据了么?如果有,那么一条更新语句应该行,不需要触发器。update
student
set
sage=sage+1
如果是插入数据时想把原来的年龄+1则需要触发器
set
ansi_nulls
on
set
quoted_identifier
on
go
alter
trigger
[dbo].[add]
on
student]
after
insert
as
begin
set
nocount
on;
update
student
set
sage=sage+1
end
默认成绩均为0是不需要触发器的,你把grade字段的默认值设为0即可。
‘玖’ sql插入触发器问题
触发器是在插入之后触发的,也就是说不管你触发器里面写什么肯定都是先插了再说(霸王硬上弓?)
然后其次,触发器里面的内容是判断选课人数是否大于180,大于的话就输入选课人数已满,那么下次别人就不能选课了,但是触发的这次是插入功成功的
‘拾’ SQL求助 插入删除修改的触发器代码编写
---创建测试表:
Create Table MyTest(
id int identity(1,1) not null primary key,
[name] varchar(100) null
)
--创建触发器:
CREATE TRIGGER trigtest--创建触发器trigtest
ON mytest--在表mytest上建
for INSERT,DELETE,UPDATE--为插入,删除,修改
AS
BEGIN
declare @a int,
@b int,
@id int,
@name varchar(100),
@oldId int,--原ID号
@oldName varchar(100)--原来的name
set @a=0
set @b=0
if exists(select * from inserted)--如果存在插入新的数据则设置@a=1
begin
set @a=1
end
if exists(select * from deleted)--如果存在删除数据则设置@b=1
begin
set @b=1
end
if (@a=1)and(@b=0)--新增:当插入表inserted存在数据而删除表deleted不存数据时,为新增操作。
begin
select @id=id,@name=name from inserted
print '增加了ID号为【'+cast(@id as varchar(5))+'】Name为【'+@name+'】的数据!'
end
if (@a=1)and(@b=1)--修改:当插入表insertedt和删除表deleted都存在数据时,为修改操作。
begin
select @id=id,@name=name from inserted
select @id=id,@oldname=name from deleted
print '修改了ID号为【'+cast(@id as varchar(5))+'】的数据Name由【'+@oldname+'】变为【'+@name+'】'
end
if (@a=0)and(@b=1)--删除:当插入表inserted不存在数据而删除表deleted存在数据时,为删除操作。
begin
select @id=id,@name=name from deleted
print '删除了ID号为【'+cast(@id as varchar(5))+'】Name为【'+@name+'】的数据!'
end
END