SQL创建触发器trigger1,实现当修改学生信息表中的某个学生的学号时,对应学生成绩表中的学号也做修改

SQL创建触发器trigger1,实现当修改学生信息表中的某个学生的学号时,对应学生成绩表中的学号也做修改,第1张

create trigger trigger1 on 学生信息表

for update

declare @old_id int

declare @new_id int

select @old_id=学号 from deleted

select @new_id=学号 from inserted

if update(学号)

begin

update 学生成绩表 set 学号=@new_id where 学号=@old_id

end

你运行下,应该没有问题。 如果你的学号不是int类型,而是varchar类型的话,修改下定义两个ID的类型。

建立两个单域的表格。一个表格中为姓名列表(表格名:data)。

另一个表格中是所插入字符的字符数(表格名:chars)。在data表格中定义一个触发器

每次在其中插入一个新姓名时,chars表格中运行的总数就会根据新插入记录的字符数目进行自动更新。

(见列表A)

mysql> CREATE TABLE data (name VARCHAR(255));

Query OK, 0 rows affected (009 sec)

mysql> CREATE TABLE chars (count INT(10));

Query OK, 0 rows affected (007 sec)

mysql> INSERT INTO chars (count) VALUES (0);

Query OK, 1 row affected (000 sec)

mysql> CREATE TRIGGER t1 AFTER INSERT ON

data FOR EACH ROW UPDATE chars SET count = count + CHAR_LENGTH(NEWname);

Query OK, 0 rows affected (001 sec)

列表A

理解上面代码的关键在于CREATE TRIGGER命令,被用来定义一个新触发器。这个命令建立一个新触发器,假定的名称为t1,每次有一个新记录插入到data表格中时,t1就被激活。

在这个触发器中有两个重要的子句:

AFTER INSERT子句表明触发器在新记录插入data表格后激活。

UPDATE chars SET count = count + CHAR_LENGTH(NEWname)子句表示触发器激活后执行的SQL命令。在本例中,该命令表明用新插入的dataname域的字符数来更新 charscount栏。这一信息可通过内置的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(NEWname)

Timing: AFTER

Created: NULL

ql_mode:

1 row in set (001 sec)

列表B

激活触发器后,开始对它进行测试。试着在data表格中插入几个记录:

mysql> INSERT INTO data (name) VALUES ('Sue'), ('Jane');

Query OK, 2 rows affected (000 sec)

Records: 2Duplicates: 0Warnings: 0

然后检查chars表格看触发器是否完成它该完成的任务:

mysql> SELECT FROM chars;

+-------+

| count |

+-------+

| 7|

+-------+

1 row in set (000 sec)

data表格中的INSERT命令激活触发器,计算插入记录的字符数,并将结果存储在chars表格中。如果往data表格中增加另外的记录,charscount值也会相应增加。

触发器应用完毕后,可有DROP TRIGGER命令轻松删除它。

mysql> DROP TRIGGER t1;

Query OK, 0 rows affected (000 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 (009 sec)

列表C

接下来,我将在accounts表格中定义一个触发器。(列表D)

mysql> CREATE TRIGGER t1 AFTER UPDATEON accounts

FOR EACH ROW INSERT INTO audit (id, balance, user, time)

VALUES (OLDid, NEWbalance, CURRENT_USER(), NOW());

Query OK, 0 rows affected (004 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 (000 sec)

表A

然后,检查触发器是否被激活:

mysql> SHOW TRIGGERS \G

1 row

Trigger: t1

Event: UPDATE

Table: accounts

Statement: INSERT INTO audit (id, balance, user, time)

VALUES (OLDid, NEWbalance, CURRENT_USER(), NOW())

Timing: AFTER

Created: NULL

Sql_mode:

1 row in set (001 sec)

再来看最后的结果(列表E):

mysql> UPDATE accounts SET balance = 500 WHERE id = 1;

Query OK, 1 row affected (000 sec)

Rows matched: 1Changed: 1Warnings: 0

mysql> UPDATE accounts SET balance = 900 WHERE id = 3;

Query OK, 1 row affected (001 sec)

Rows matched: 1Changed: 1Warnings: 0

mysql> UPDATE accounts SET balance = 1900 WHERE id = 1;

Query OK, 1 row affected (000 sec)

Rows matched: 1Changed: 1Warnings: 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 (000 sec)

给表B建立一个Insert触发器,当表B发生插入记录的时候自动给表A的B字段+1

CREATE TRIGGER INSERT_B ON B

FOR INSERT

AS

至于具体中间的业务你怎么写就自己考虑啦

对了在触发器中可以调用虚拟的临时表inserted,这个是在插入触发器里面可以调用的,目的是查询在插入时某些值是否符合要求或者什么业务逻辑是否满足之类的等等

这个触发器只要表B发生了插入 *** 作就会执行,如果你单纯只想给表A在表B有插入的时候+1

那么直接就UPDATE表A的B字段+1就好了

楼主的语法是按orcale来的吧?sql server2000中的trigger中没有before、for each row、new等关键字。下面我写了个,能实现你的功能,只是不知道有没有性能更好的写法。期待更佳的

if exists(select from sysobjects where name='Insert_Or_Update_sal')

drop trigger Insert_Or_Update_sal

go

CREATE TRIGGER Insert_Or_Update_sal

ON TEACHER

FOR insert,update

AS

BEGIN

declare @job char(8),@sal numeric(7,2),@eno numeric(4)

--inserted是insert或update触发表临时表,参考下面帮助。

select @eno=eno,@job=job,@sal=sal from inserted

if @job='教授' and @sal<4000

update teacher set sal=4000 where eno=@eno

--下面的两句也可能完成这个功能,只是性能低一些。

-- if exists(select from inserted where job='教授' and sal< 4000)

-- update TEACHER set sal=4000 where job='教授' AND sal<4000

END

下面是sql server2000帮助文档中相关trigger的部分,希望对你有帮助。

----------------------------------------------------------------

CREATE TRIGGER

创建触发器,触发器是一种特殊的存储过程,在用户试图对指定的表执行指定的数据修改语句时自动执行。Microsoft® SQL Server™ 允许为任何给定的 INSERT、UPDATE 或 DELETE 语句创建多个触发器。

语法

CREATE TRIGGER trigger_name

ON { table | view }

[ WITH ENCRYPTION ]

{

{ { FOR | AFTER | INSTEAD OF } { [ INSERT ] [ , ] [ UPDATE ] }

[ WITH APPEND ]

[ NOT FOR REPLICATION ]

AS

[ { IF UPDATE ( column )

[ { AND | OR } UPDATE ( column ) ]

[ n ]

| IF ( COLUMNS_UPDATED ( ) { bitwise_operator } updated_bitmask )

{ comparison_operator } column_bitmask [ n ]

} ]

sql_statement [ n ]

}

}

参数

trigger_name

是触发器的名称。触发器名称必须符合标识符规则,并且在数据库中必须唯一。可以选择是否指定触发器所有者名称。

Table | view

是在其上执行触发器的表或视图,有时称为触发器表或触发器视图。可以选择是否指定表或视图的所有者名称。

WITH ENCRYPTION

加密 syscomments 表中包含 CREATE TRIGGER 语句文本的条目。使用 WITH ENCRYPTION 可防止将触发器作为 SQL Server 复制的一部分发布。

AFTER

指定触发器只有在触发 SQL 语句中指定的所有 *** 作都已成功执行后才激发。所有的引用级联 *** 作和约束检查也必须成功完成后,才能执行此触发器。

如果仅指定 FOR 关键字,则 AFTER 是默认设置。

不能在视图上定义 AFTER 触发器。

INSTEAD OF

指定执行触发器而不是执行触发 SQL 语句,从而替代触发语句的 *** 作。

在表或视图上,每个 INSERT、UPDATE 或 DELETE 语句最多可以定义一个 INSTEAD OF 触发器。然而,可以在每个具有 INSTEAD OF 触发器的视图上定义视图。

INSTEAD OF 触发器不能在 WITH CHECK OPTION 的可更新视图上定义。如果向指定了 WITH CHECK OPTION 选项的可更新视图添加 INSTEAD OF 触发器,SQL Server 将产生一个错误。用户必须用 ALTER VIEW 删除该选项后才能定义 INSTEAD OF 触发器。

{ [DELETE] [,] [INSERT] [,] [UPDATE] }

是指定在表或视图上执行哪些数据修改语句时将激活触发器的关键字。必须至少指定一个选项。在触发器定义中允许使用以任意顺序组合的这些关键字。如果指定的选项多于一个,需用逗号分隔这些选项。

对于 INSTEAD OF 触发器,不允许在具有 ON DELETE 级联 *** 作引用关系的表上使用 DELETE 选项。同样,也不允许在具有 ON UPDATE 级联 *** 作引用关系的表上使用 UPDATE 选项。

WITH APPEND

指定应该添加现有类型的其它触发器。只有当兼容级别是 65 或更低时,才需要使用该可选子句。如果兼容级别是 70 或更高,则不必使用 WITH APPEND 子句添加现有类型的其它触发器(这是兼容级别设置为 70 或更高的 CREATE TRIGGER 的默认行为)。有关更多信息,请参见 sp_dbcmptlevel。

WITH APPEND 不能与 INSTEAD OF 触发器一起使用,或者,如果显式声明 AFTER 触发器,也不能使用该子句。只有当出于向后兼容而指定 FOR 时(没有 INSTEAD OF 或 AFTER),才能使用 WITH APPEND。以后的版本将不支持 WITH APPEND 和 FOR(将被解释为 AFTER)。

NOT FOR REPLICATION

表示当复制进程更改触发器所涉及的表时,不应执行该触发器。

AS

是触发器要执行的 *** 作。

sql_statement

是触发器的条件和 *** 作。触发器条件指定其它准则,以确定 DELETE、INSERT 或 UPDATE 语句是否导致执行触发器 *** 作。

当尝试 DELETE、INSERT 或 UPDATE *** 作时,Transact-SQL语句中指定的触发器 *** 作将生效。

触发器可以包含任意数量和种类的 Transact-SQL 语句。触发器旨在根据数据修改语句检查或更改数据;它不应将数据返回给用户。触发器中的 Transact-SQL 语句常常包含控制流语言。CREATE TRIGGER 语句中使用几个特殊的表:

deleted 和 inserted 是逻辑(概念)表。这些表在结构上类似于定义触发器的表(也就是在其中尝试用户 *** 作的表);这些表用于保存用户 *** 作可能更改的行的旧值或新值。例如,若要检索 deleted 表中的所有值,请使用:

SELECT

FROM deleted

如果兼容级别等于 70,那么在 DELETE、INSERT 或 UPDATE 触发器中,SQL Server 将不允许引用 inserted 和 deleted 表中的 text、ntext 或 image 列。不能访问 inserted 和 deleted 表中的 text、ntext 和 image 值。若要在 INSERT 或 UPDATE 触发器中检索新值,请将 inserted 表与原始更新表联接。当兼容级别是 65 或更低时,对 inserted 或 deleted 表中允许空值的text、ntext 或 image 列,将返回空值;如果这些列不可为空,则返回零长度字符串。

当兼容级别是 80 或更高时,SQL Server 允许在表或视图上通过 INSTEAD OF 触发器更新 text、ntext 或 image 列。

n

是表示触发器中可以包含多条 Transact-SQL 语句的占位符。对于 IF UPDATE (column) 语句,可以通过重复 UPDATE (column) 子句包含多列。

IF UPDATE (column)

测试在指定的列上进行的 INSERT 或 UPDATE *** 作,不能用于 DELETE *** 作。可以指定多列。因为在 ON 子句中指定了表名,所以在 IF UPDATE 子句中的列名前不要包含表名。若要测试在多个列上进行的 INSERT 或 UPDATE *** 作,请在第一个 *** 作后指定单独的 UPDATE(column) 子句。在 INSERT *** 作中 IF UPDATE 将返回 TRUE 值,因为这些列插入了显式值或隐性 (NULL) 值。

说明 IF UPDATE (column) 子句的功能等同于 IF、IFELSE 或 WHILE 语句,并且可以使用 BEGINEND 语句块。有关更多信息,请参见控制流语言。

可以在触发器主体中的任意位置使用 UPDATE (column)。

column

是要测试 INSERT 或 UPDATE *** 作的列名。该列可以是 SQL Server 支持的任何数据类型。但是,计算列不能用于该环境中。有关更多信息,请参见数据类型。

IF (COLUMNS_UPDATED())

测试是否插入或更新了提及的列,仅用于 INSERT 或 UPDATE 触发器中。COLUMNS_UPDATED 返回 varbinary 位模式,表示插入或更新了表中的哪些列。

COLUMNS_UPDATED 函数以从左到右的顺序返回位,最左边的为最不重要的位。最左边的位表示表中的第一列;向右的下一位表示第二列,依此类推。如果在表上创建的触发器包含 8 列以上,则 COLUMNS_UPDATED 返回多个字节,最左边的为最不重要的字节。在 INSERT *** 作中 COLUMNS_UPDATED 将对所有列返回 TRUE 值,因为这些列插入了显式值或隐性 (NULL) 值。

可以在触发器主体中的任意位置使用 COLUMNS_UPDATED。

bitwise_operator

是用于比较运算的位运算符。

updated_bitmask

是整型位掩码,表示实际更新或插入的列。例如,表 t1 包含列 C1、C2、C3、C4 和 C5。假定表 t1 上有 UPDATE 触发器,若要检查列 C2、C3 和 C4 是否都有更新,指定值 14;若要检查是否只有列 C2 有更新,指定值 2。

comparison_operator

是比较运算符。使用等号 (=) 检查 updated_bitmask 中指定的所有列是否都实际进行了更新。使用大于号 (>) 检查 updated_bitmask 中指定的任一列或某些列是否已更新。

column_bitmask

是要检查的列的整型位掩码,用来检查是否已更新或插入了这些列。

注释

触发器常常用于强制业务规则和数据完整性。SQL Server 通过表创建语句(ALTER TABLE 和 CREATE TABLE)提供声明引用完整性 (DRI);但是 DRI 不提供数据库间的引用完整性。若要强制引用完整性(有关表的主键和外键之间关系的规则),请使用主键和外键约束(ALTER TABLE 和 CREATE TABLE 的 PRIMARY KEY 和 FOREIGN KEY 关键字)。如果触发器表存在约束,则在 INSTEAD OF 触发器执行之后和 AFTER 触发器执行之前检查这些约束。如果违反了约束,则回滚 INSTEAD OF 触发器 *** 作且不执行(激发)AFTER 触发器。

可用 sp_settriggerorder 指定表上第一个和最后一个执行的 AFTER 触发器。在表上只能为每个 INSERT、UPDATE 和 DELETE *** 作指定一个第一个执行和一个最后一个执行的 AFTER 触发器。如果同一表上还有其它 AFTER 触发器,则这些触发器将以随机顺序执行。

如果 ALTER TRIGGER 语句更改了第一个或最后一个触发器,则将除去已修改触发器上设置的第一个或最后一个特性,而且必须用 sp_settriggerorder 重置排序值。

只有当触发 SQL 语句(包括所有与更新或删除的对象关联的引用级联 *** 作和约束检查)成功执行后,AFTER 触发器才会执行。AFTER 触发器检查触发语句的运行效果,以及所有由触发语句引起的 UPDATE 和 DELETE 引用级联 *** 作的效果。

触发器限制

CREATE TRIGGER 必须是批处理中的第一条语句,并且只能应用到一个表中。

触发器只能在当前的数据库中创建,不过触发器可以引用当前数据库的外部对象。

如果指定触发器所有者名称以限定触发器,请以相同的方式限定表名。

在同一条 CREATE TRIGGER 语句中,可以为多种用户 *** 作(如 INSERT 和 UPDATE)定义相同的触发器 *** 作。

如果一个表的外键在 DELETE/UPDATE *** 作上定义了级联,则不能在该表上定义 INSTEAD OF DELETE/UPDATE 触发器。

在触发器内可以指定任意的 SET 语句。所选择的 SET 选项在触发器执行期间有效,并在触发器执行完后恢复到以前的设置。

与使用存储过程一样,当触发器激发时,将向调用应用程序返回结果。若要避免由于触发器激发而向应用程序返回结果,请不要包含返回结果的 SELECT 语句,也不要包含在触发器中进行变量赋值的语句。包含向用户返回结果的 SELECT 语句或进行变量赋值的语句的触发器需要特殊处理;这些返回的结果必须写入允许修改触发器表的每个应用程序中。如果必须在触发器中进行变量赋值,则应该在触发器的开头使用 SET NOCOUNT 语句以避免返回任何结果集。

DELETE 触发器不能捕获 TRUNCATE TABLE 语句。尽管 TRUNCATE TABLE 语句实际上是没有 WHERE 子句的 DELETE(它删除所有行),但它是无日志记录的,因而不能执行触发器。因为 TRUNCATE TABLE 语句的权限默认授予表所有者且不可转让,所以只有表所有者才需要考虑无意中用 TRUNCATE TABLE 语句规避 DELETE 触发器的问题。

无论有日志记录还是无日志记录,WRITETEXT 语句都不激活触发器。

存储过程:

mysql

>

CREATE

PROCEDURE

up_sbwcsl(IN

xbh

int)

BEGIN

begin

select

科研计划系部号,是否申报,完成数量

from

科研计划

Where

科研计划系部号=@xbh

and

是否申报='是'

END;

触发器语法一样的。

Create

trigger

up_ysbz

On

科研计划

For

update

As

Begin

If

update(验收标志)

Update

科研计划

set

验收标志='验收通过'

end

以上就是关于SQL创建触发器trigger1,实现当修改学生信息表中的某个学生的学号时,对应学生成绩表中的学号也做修改全部的内容,包括:SQL创建触发器trigger1,实现当修改学生信息表中的某个学生的学号时,对应学生成绩表中的学号也做修改、mysql数据库触发器,怎么做满足条件触发、sql 触发器 修改值等相关内容解答,如果想了解更多相关内容,可以关注我们,你们的支持是我们更新的动力!

欢迎分享,转载请注明来源:内存溢出

原文地址:https://54852.com/sjk/10184294.html

(0)
打赏 微信扫一扫微信扫一扫 支付宝扫一扫支付宝扫一扫
上一篇 2023-05-06
下一篇2023-05-06

发表评论

登录后才能评论

评论列表(0条)

    保存