前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Mysql 触发器基础

Mysql 触发器基础

作者头像
爱知汇
发布2020-10-10 10:06:05
8.2K0
发布2020-10-10 10:06:05
举报
文章被收录于专栏:网页开发/美化网页开发/美化

触发器的概念

触发器(trigger)是MySQL提供给程序员和数据分析员来保证数据完整性的一种方法,它是与表事件相关的特殊的存储过程,它的执行不是由程序调用,也不是手工启动,而是由事件来触发,比如当对一个表进行操作(insert,delete, update)时就会激活它执行。——百度百科

上面是百度给的触发器的概念,我理解的触发器的概念,就是你执行一条sql语句,这条sql语句的执行会自动去触发执行其他的sql语句,就这么简单。

超简说明:sql1->触发->sqlN,一条sql触发多个sql

触发器创建的四个要素

  1. 监视地点(table)
  2. 监视事件(insert/update/delete)
  3. 触发时间(after/before)
  4. 触发事件(insert/update/delete)

创建触发器

需求:在下订单的时候,对应的商品的库存量要相应的减少,即买几个商品就减少多少个库存量。

订单表:ord 商品表:goods

首先来创建表并添加几条数据:

代码语言:javascript
复制
create table goods(    gid int,    name varchar(20),    num smallint);create table ord(    oid int,    gid int,    much smallint);insert into goods values(1,'cat',40);insert into goods values(2,'dog',63);insert into goods values(3,'pig',87);

然后按照触发器创建的四个要素来进行分析:

监视谁:ord(订单表) 监视动作:insert(插入操作) 触发时间:after(在插入操作后触发) 触发事件:update(触发更新操作)

最后创建触发器:

代码语言:javascript
复制
create trigger t1 afterinsert on ordfor each rowbegin  update goods set num=num-2 where gid = 1;end$

分析:触发器的名称为t1,触发时间为after,监视动作为insert,监视ord表,for each row最后在进行讨论,这里先记住就行了,begin和end之间写触发事件,这里是一个update语句。意思是不论我下什么订单,都会把商品编号为1的商品的库存量减去2个。

注意:先不要运行上面的代码,因为mysql的执行结束标识默认是;。如果运行以上的sql语句,mysql碰到;时会自动停止执行,然后end语句就执行不到了。所以我们需要先将mysql的结束标识符改为其他的字符,一般都选用或者,这里选用来作为执行的结束标识。使用下面的语句来修改MySQL执行的结束标识。

代码语言:javascript
复制
delimiter $ //设置MySQL执行结束标志,默认为;

查看和删除已有的触发器

  1. 查看已有触发器:show triggers
  2. 删除已有触发器:drop trigger triggerName

触发器中引用行变量

  1. 在触发目标上执行insert操作后会有一个新行,如果在触发事件中需要用到这个新行的变量,可以用new关键字表示
  2. 在触发目标上执行delete操作后会有一个旧行,如果在触发事件中需要用到这个旧行的变量,可以用old关键字表示
  3. 在触发目标上执行update操作后原纪录是旧行,新记录是新行,可以使用new和old关键字来分别操作

当下订单时减少相应的货品的库存量,创建触发器:

代码语言:javascript
复制
create trigger t2afterinsert on ordfor each rowbegin  update goods set num=num-new.much where gid=new.gid;end$

当删除订单时增加相应的修改货品的库存量,创建触发器:

代码语言:javascript
复制
create trigger t3afterdeleteon ordfor each rowbegin  update goods set num=num+old.much where gid=old.gid;end$

当更新订单的购买数修改相应的修改货品的库存量,创建触发器:

代码语言:javascript
复制
create trigger t4before updateon ordfor each rowbegin  update goods set num=num+old.much-new.much where gid = new.gid;end$

after和before的区别

after操作,是在执行了监视动作后,才会执行触发事件 before操作,是在执行了监视动作前,会执行触发事件 两者在一般的触发器中并没有什么区别,但是有的时候有区别,如:

需求:在用户定了超过库存的订单后,会修改该订单的订购数量,使订购数量的最大值和库存量相同 分析:首先判断 订购量 > 库存量,然后做将订购量改为库存量

创建触发器:

代码语言:javascript
复制
create trigger t5beforeinsert on ordfor each rowbegin  declare restNum int;  select num into restNum from goods where gid = new.gid;  if new.much > restNum then     set new.much = restNum;  end if;  update goods set num=num-new.much where gid=new.gid;end$

注意:这里如果使用的是after就会报错,如果使用的是after,就会先执行insert操作,也就是插入订单操作,然后在进行判断下单数量和库存量,得出新的下单数量,可是已经执行了下单操作了,所以就会报错。这里必须使用before操作。

for each row是干什么的?

在oracle触发器中,触发器分为行触发器和语句触发器

比如:

代码语言:javascript
复制
create trigger tnafterupdateon xxtablefor each row #每一行受影响,触发事件都执行,叫做行触发器begin  sqlN;end$

执行:

代码语言:javascript
复制
update xxtable set xxx=xxx where id>100;

该修改操作假设100行,那么sqlN,会触发多少次?答案:会触发100次。

拓展:

在oracle中,for each row如果不写,无论update语句一次影响了多少行,都只执行一次触发事件。 比如:1人下了订单,买了5件商品,insert 5次,可以用行级触发器,修改5次库存;用语句级触发器触发,insert一条发货提醒。 遗憾的是mysql目前不支持语句级触发器。

本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2020-09-25 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
相关产品与服务
云数据库 SQL Server
腾讯云数据库 SQL Server (TencentDB for SQL Server)是业界最常用的商用数据库之一,对基于 Windows 架构的应用程序具有完美的支持。TencentDB for SQL Server 拥有微软正版授权,可持续为用户提供最新的功能,避免未授权使用软件的风险。具有即开即用、稳定可靠、安全运行、弹性扩缩等特点。
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档