mysql数据库触发器,怎么做满足条件触发

发布网友

我来回答

3个回答

懂视网

1.触发器:

什么是触发器?我们什么时候能够使用触发器?

  触发器就是用来监听某个表的变化,当这个表发生变化的时候来触发某种操作..比若说两个表是相互关联的,当我们在对其中一个表格进行操作的同时,另一个表内的数据也需要进行某种操作,那么我们就需要建立一个触发器来监听任何一个表发生的变化,当其中一个表的数据发生变化的同时,触发器里的代码块将会对另一个表格进行对数据的某种操作。。

总而言之,触发器一般是使用在表与表之间的,单个的表建立触发器是没有任何的意义的。。。。

触发器的监听范围:增,删,改

触发的操作:增,删,改

  有点懵,我们来举一个实例。。。比如说一个订餐系统。。其中存在着一个订单数量的数据库,一个外卖样式的数据库。。当客户进行订餐的时候,订单的数量变化的同时,那么外卖样式的数据库信息也要发生变化,这两个表是相互关联的。。总不能我们的库存外卖数量已经空了,我们还能够接受订单吧?一旦出现这种情况,那么必定会出现问题。。。因此我们可以使用触发器来解决这个问题。。。

 

create table order_form
(
 order_form_id int not null,  //每一种外卖的id
 order_form_much int not null //每一种外卖允许订的最大数量
); //建立一个订单表格.....
create table food
(
 food_id int not null,   //外卖的id信息
 food_name nvarchar(50) not null, //外卖的名字,样式
 food_much int not null  //外卖的库存量 
);

insert into order_form values(1,10),(2,5),(3,15),(4,20),(5,5); //插入数据信息

insert into food values(1,‘澳洲鲍鱼‘,10),(2,‘龙虾‘,5),(3,‘鱼翅‘,15),(4,‘花枝‘,20),(5,‘干贝‘,5); //插入信息

比如说用户订了一份澳洲鲍鱼,那么food内名字为澳洲鲍鱼的库存数量就需要减一,同时订单内名字为澳洲鲍鱼的数量也需要减一。。。当最大数量减为0的时候,那么就代表库存空了,那么就不再接受客户的订单了....

这个过程的实现需要触发器。。。基本语法:
create trigger trigger_name (after/before insert/update/delete 监听的范围) on table_name for each row 
begin 
 一句或者多句sql语句;
end;

create trigger trg1 after on order_form for each row
begin 
 update food set food_much=food_much-1 where food_id=old.order_form_id; //这个old必须要加,否则出错....
end;

update order_form set order_form_much=order_form_much-1 where order_form_id=1; //当这句话执行的时候触发器将被触发。。意思就是客户定了一份外卖,那么订单数在减一的同时,库存量同时也要减少....

这个触发器不灵活,因为客户不可能每次都定一份外卖吧。。有可能订多份。。。我们是可以对订的数量进行判断,但是如果订了1000份,我们总不能判断1000次吧。。。因此我们还可以新建一个表格,里面保存着每一份外卖被订了多少次。。然后通过对这个表保存的信息来修改其他两个表信息....

create table c_count
(
 id int not null,
 num int not null default 0,
 primary key(id);
); 
当有订单的时候,我们把订单的数据更新到这个表格内。。

建立触发器。。。
create trigger trg2 after on update on for each row 
begin 
 update food set food_much=food_much-old.num where food_id=old.id;
 update food set order_form_much=order_form_much-old.num where order_form_id=old.id;
end;

update c_count set num=2 where id=3;//触发器被触发。。。

 

  上面只是简单的介绍了一下更新操作,我再简单的说一下插入和删除操作是如何触发监听的,虽然说模式都差不多,但是还是有一些需要注意的地方...

删除操作后的触发:

比如说当用户取消订单的时候,那么我们对应的允许的订单数量和库存量也要随之发生改变。。。

 

create table c_count
(
 id int not null,
 num int not null default 0,
 primary key(id);
); //还是那个表格。。。

create trigger trg2 after on update on for each row 
begin 
 update food set food_much=food_much+old.num where food_id=old.id;
 update food set order_form_much=order_form_much+old.num where order_form_id=old.id; //也是使用(old.数据) 来表示删除前的数据信息
end;

delete from c_count where id=1; 

 

插入操作的触发:

比如说:店内新增了一样产品。。。那么我们仍然需要更新数据库内的信息...

 

mysql> create trigger trg1 after insert on c_count for each row 
 -> begin 
 -> insert into food values(6,‘比目鱼‘,8);
 -> insert into order_form values(6,8);
 -> end;

mysql> insert into c_count values(6,8); 

//这里还有一个注意。。就是像上面删除的地方我们需要使用(old.数据)来操作我们删除之前的数据。。。那么这里的插入操作,我们需要使用(new.数据)来操作我们插入的新数据...

删除触发器:

drop trigger trg1;

查看触发器信息:

show triggers;

触发器里after和before的区别:
after是先安完成数据的增,删,改再触发
触发的语句晚于监视的增,删,改,无法影响前面的增删改动作
before是先完成触发,再增删改
触发的语句先于监视的增删改发生,我们有机会判断,修改即将发生的操作

 

Mysql学习笔记(十二)触发器

标签:

热心网友

建立两个单域的表格。一个表格中为姓名列表(表格名:data)。
另一个表格中是所插入字符的字符数(表格名:chars)。在data表格中定义一个触发器。
每次在其中插入一个新姓名时,chars表格中运行的总数就会根据新插入记录的字符数目进行自动更新。
(见列表A)
mysql> CREATE TABLE data (name VARCHAR(255));
Query OK, 0 rows affected (0.09 sec)
mysql> CREATE TABLE chars (count INT(10));
Query OK, 0 rows affected (0.07 sec)
mysql> INSERT INTO chars (count) VALUES (0);
Query OK, 1 row affected (0.00 sec)
mysql> CREATE TRIGGER t1 AFTER INSERT ON
data FOR EACH ROW UPDATE chars SET count = count + CHAR_LENGTH(NEW.name);
Query OK, 0 rows affected (0.01 sec)
列表A
理解上面代码的关键在于CREATE TRIGGER命令,被用来定义一个新触发器。这个命令建立一个新触发器,假定的名称为t1,每次有一个新记录插入到data表格中时,t1就被激活。
在这个触发器中有两个重要的子句:
AFTER INSERT子句表明触发器在新记录插入data表格后激活。
UPDATE chars SET count = count + CHAR_LENGTH(NEW.name)子句表示触发器激活后执行的SQL命令。在本例中,该命令表明用新插入的data.name域的字符数来更新 chars.count栏。这一信息可通过内置的MySQL函数CHAR_LENGTH()获得。
放在源表格域名前面的NEW关键字也值得注意。这个关键字表明触发器应考虑域的new值(也就是说,刚被插入到域中的值)。MySQL还支持相应的OLD前缀,可用它来指域以前的值。
可以通过调用SHOW TRIGGER命令来检查触发器是否被激活,如列表B所示。
mysql> SHOW TRIGGERS\G
*************************** 1. row ***************************
?Trigger: t1
?Event: INSERT
?Table: data
Statement: UPDATE chars SET count = count + CHAR_LENGTH(NEW.name)
Timing: AFTER
?Created: NULL
ql_mode:
1 row in set (0.01 sec)
列表B
激活触发器后,开始对它进行测试。试着在data表格中插入几个记录:
mysql> INSERT INTO data (name) VALUES ('Sue'), ('Jane');
Query OK, 2 rows affected (0.00 sec)
Records: 2?Duplicates: 0?Warnings: 0
然后检查chars表格看触发器是否完成它该完成的任务:
mysql> SELECT * FROM chars;
+-------+
| count |
+-------+
| 7|
+-------+
1 row in set (0.00 sec)
data表格中的INSERT命令激活触发器,计算插入记录的字符数,并将结果存储在chars表格中。如果往data表格中增加另外的记录,chars.count值也会相应增加。
触发器应用完毕后,可有DROP TRIGGER命令轻松删除它。
mysql> DROP TRIGGER t1;
Query OK, 0 rows affected (0.00 sec)
注意:理想情况下,你还需要一个倒转触发器,每当一个记录从源表格中删除时,它从字符总数中减去记录的字符数。这很容易做到,你可以把它当作练习来完成。提示:应用BEFORE DELETE ON子句是其中一种方法。
现在,要建立一个审计记录来追踪对这个表格所做的改变。这个记录将反映表格的每项改变,并向用户说明由谁做出改变以及改变的时间。需要建立一个新表格来存储这一信息(表格名:audit),如下所示。(列表C)
mysql> CREATE TABLE audit (id INT(7), balance FLOAT, user VARCHAR(50)
NOT NULL, time TIMESTAMP NOT NULL);
Query OK, 0 rows affected (0.09 sec)
列表C
接下来,我将在accounts表格中定义一个触发器。(列表D)
mysql> CREATE TRIGGER t1 AFTER UPDATEON accounts
FOR EACH ROW INSERT INTO audit (id, balance, user, time)
VALUES (OLD.id, NEW.balance, CURRENT_USER(), NOW());
Query OK, 0 rows affected (0.04 sec)
列表D
要是已经走到这一步,就很容易理解。accounts表格每经历一次UPDATE,触发器插入(INSERT)对应记录的id、新的余额、当前时间和登录audit表格的用户的名称。
实现中的例子:用触发器审计记录
既然了触发器的基本原理,来看一个稍稍复杂的例子。常用触发器来建立一个自动“审计记录”,以记录各种用户对数据库的更改。为了解审计记录的实际应用,请看下面的表格(表格名:accounts),它列出了一个用户的三个银行账户余额。(表A)
mysql> SELECT * FROM accounts;
+----+------------+---------+
| id | label| balance |
+----+------------+---------+
|1 | Savings #1 |500 |
|2 | Current #1 |2000 |
|3 | Current #2 |3500 |
+----+------------+---------+
3 rows in set (0.00 sec)
表A
然后,检查触发器是否被激活:
mysql> SHOW TRIGGERS \G
*************************** 1. row ***************************
?Trigger: t1
?Event: UPDATE
?Table: accounts
Statement: INSERT INTO audit (id, balance, user, time)
VALUES (OLD.id, NEW.balance, CURRENT_USER(), NOW())
Timing: AFTER
?Created: NULL
Sql_mode:
1 row in set (0.01 sec)
再来看最后的结果(列表E):
mysql> UPDATE accounts SET balance = 500 WHERE id = 1;
Query OK, 1 row affected (0.00 sec)
Rows matched: 1?Changed: 1?Warnings: 0
mysql> UPDATE accounts SET balance = 900 WHERE id = 3;
Query OK, 1 row affected (0.01 sec)
Rows matched: 1?Changed: 1?Warnings: 0
mysql> UPDATE accounts SET balance = 1900 WHERE id = 1;
Query OK, 1 row affected (0.00 sec)
Rows matched: 1?Changed: 1?Warnings: 0
列表E
注意,对accounts表格所作的改变已被记录到audit表格中,将来如果出现问题,可以方便地从中进行恢复。
mysql> SELECT * FROM audit;
+------+---------+----------------+---------------------+
| id| balance | user| time|
+------+---------+----------------+---------------------+
|1 |500 | root@localhost | 2006-04-22 12:52:15 |
|3 |900 | root@localhost | 2006-04-22 12:53:15 |
|1 |1900 | root@localhost | 2006-04-22 12:53:23 |
+------+---------+----------------+---------------------+
3 rows in set (0.00 sec)

热心网友

语法错误。


MySQL 里面。  

IF   条件  THEN   

  ......

END IF

的。


例如:

mysql> DELIMITER //
mysql> CREATE PROCEDURE TestIfElse
    -> (
    ->   p_val  INT
    -> )
    -> BEGIN
    ->   IF (p_val = 1) THEN
    ->     SELECT '1' AS A;
    ->   ELSEIF (p_val = 2) THEN
    ->     SELECT '2' AS A;
    ->   ELSE
    ->     SELECT 'other' AS A;
    ->   END IF;
    -> END//
Query OK, 0 rows affected (0.05 sec)

mysql> call TestIfElse(1);
    -> //
+---+
| A |
+---+
| 1 |
+---+
1 row in set (0.00 sec)

Query OK, 0 rows affected (0.00 sec)

mysql> call TestIfElse(2);
    -> //
+---+
| A |
+---+
| 2 |
+---+
1 row in set (0.00 sec)

Query OK, 0 rows affected (0.00 sec)

mysql> call TestIfElse(3);
    -> //
+-------+
| A     |
+-------+
| other |
+-------+
1 row in set (0.00 sec)

Query OK, 0 rows affected (0.00 sec)

mysql>



你那里少了一个   THEN

追问


我已经做出来了!谢谢你的回复!以上是正确语句!和你说的一样哈!谢谢!

声明声明:本网页内容为用户发布,旨在传播知识,不代表本网认同其观点,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。E-MAIL:11247931@qq.com