【SQL Server】数据库开发指南(九)详细讲解 MS-SQL 触发器的基础概念与应用场景

news2024/11/24 5:32:55

本系列博文还在更新中,收录在专栏:#MS-SQL Server 专栏中。

本系列文章列表如下:

【SQL Server】 Linux 运维下对 SQL Server 进行安装、升级、回滚、卸载操作
【SQL Server】数据库开发指南(一)数据库设计的核心概念和基本步骤
【SQL Server】数据库开发指南(二)MSSQL数据库开发对于库、表、数据类型、约束等相关操作
【SQL Server】数据库开发指南(三)面向数据分析的 T-SQL 编程技巧与实践
[ 云原生 | Docker ] 构建高可用性的 SQL Server:Docker 容器下的主从同步实现指南
【SQL Server】数据库开发指南(五)T-SQL 高级查询综合应用与实战
【SQL Server】数据库开发指南(六)索引和视图的使用技巧、方法与综合应用
【SQL Server】数据库开发指南(七)MS-SQL存储过程全面解析:种类、优点和创建方法详解
【SQL Server】数据库开发指南(八)高级数据处理技术 MS-SQL 事务、异常和游标的深入研究
【SQL Server】数据库开发指南(九)详细讲解 MS-SQL 触发器的基础概念与应用场景

文章目录

    • 前言
    • 一、什么是触发器
      • 1.1 DML 触发器
        • 1.1.1 Instead of 触发器
        • 1.1.2 After 触发器
      • 1.2 DDL 触发器
      • 1.3 INSERTED 表 与 DELETED 表
    • 二、T-SQL语句创建触发器各示例
      • 2.1 基本通用语法
      • 2.2 创建 insert 类型触发器
      • 2.3 创建 delete 类型触发器
      • 2.4 创建 update 类型触发器
      • 2.5 update 更新列级触发器
      • 2.6 创建 instead of 触发器
    • 三、修改触发器
    • 四、启用、禁用触发器
    • 五、查询创建的触发器信息
    • 六、结合触发器操作日志示例

前言

本文深入探讨数据库触发器的各个方面,帮助读者更好地理解和应用触发器技术。我们将详细介绍触发器的概念和作用,并重点讨论 DML 触发器和 DDL 触发器。在 DML 触发器部分,我们将关注 Instead of 触发器和 After 触发器的用法,同时介绍 INSERTED 表和 DELETED 表的特点。

接下来,我们将提供丰富的示例,演示使用 T-SQL 语句创建不同类型的触发器,包括 insert、delete、update 类型和列级触发器。此外,我们还将讨论修改触发器的技巧、启用和禁用触发器的方法,以及查询创建的触发器信息。通过结合实际案例,我们将帮助读者更好地理解触发器的应用场景和优化策略。无论您是初学者还是有一定经验的开发者,本文都将为您提供实用的触发器知识和技巧,助您在数据库操作中更加灵活和高效。

一、什么是触发器

触发器(Trigger)是一种特殊的代码段或操作,它会在特定事件发生时自动执行。触发器通常与数据库管理系统(DBMS)相关联,用于在数据库中定义和执行自动化操作。

触发器可以根据定义的条件和事件,触发预定的动作或响应。当满足触发器定义的条件时,触发器会自动触发,并执行与之关联的代码逻辑。触发器通常用于实现数据完整性约束、自动化数据更新和审计跟踪等功能。

SQL Server 中触发器可以分为两类:DML 触发器DDL 触发器

在这里插入图片描述

1.1 DML 触发器

DML触发器(Data Manipulation Language Trigger):这种触发器与表的数据操作(插入、更新、删除)相关。当执行与表相关的 DML语句时(如:INSERT、UPDATE、DELETE),系统会自动调用执行与该表相关的 DML 触发器。这些触发器允许你在数据操作之前或之后执行自定义的逻辑。它们通常用于实现复杂的数据完整性约束、数据验证、审计跟踪等功能。

DML触发器分为: Instead of 触发器(之前触发)After 触发器(之后触发)

1.1.1 Instead of 触发器

Instead of 触发器用于替代引起触发器执行的T-SQL语句。除表之外,Instead of 触发器也可以用于视图,用来扩展视图可以支持的更新操作。当在表上执行与触发器相关的操作(如插入、更新、删除)时,如果存在适当的 INSTEAD OF 触发器,系统将跳过原始操作,而是执行 INSTEAD OF 触发器中定义的逻辑。

INSTEAD OF 触发器常见的应用:

视图更新:当操作涉及到视图时,可以使用 INSTEAD OF 触发器来处理视图的更新逻辑。视图是虚拟的表,可能包含来自多个基本表的数据。通过定义 INSTEAD OF 触发器,您可以在对视图进行插入、更新或删除操作时,自定义处理数据的逻辑。

复杂约束处理:某些复杂的数据约束无法直接通过常规的数据库约束(如主键、外键、唯一性约束)来实现。在这种情况下,您可以使用 INSTEAD OF 触发器来处理和验证数据的约束逻辑。

自定义操作:INSTEAD OF 触发器还可以用于执行自定义操作,而不仅仅是数据操作。您可以在触发器中执行复杂的业务逻辑、数据转换、数据合并等操作。

1.1.2 After 触发器

After 触发器要求只有执行某一操作 insert、update、delete 之后触发器才被触发,且只能定义在表上。After 触发器通常用于在数据操作完成后执行后续的操作或触发其他事件。

这里根据个人经验,总结几个 After 触发器的要点:

触发时机:After 触发器在执行与其关联的数据操作(如插入、更新、删除)之后触发。在执行原始操作后,触发器允许您在处理完成的数据上执行进一步的操作。

访问被触发表的数据:After 触发器可以通过使用特殊表 INSERTEDDELETED 来访问触发操作后的数据。INSERTED 表存储已插入或已更新的新数据,而 DELETED 表存储已更新或已删除的旧数据(后面会介绍)。

多个触发器的顺序:当多个 After 触发器与同一数据操作相关联时,它们将按照创建顺序依次触发。您可以使用 sp_settriggerorder 存储过程来更改 After 触发器的执行顺序。

功能和应用:After 触发器可用于执行各种后续操作,例如:

  • 更新其他相关表中的数据,以确保数据的一致性。
  • 生成日志记录或审计跟踪信息,以便记录操作历史或进行审计。
  • 触发其他触发器或存储过程,以便执行进一步的业务逻辑。
  • 发送通知或触发事件,以便在操作完成后通知相关方。

注意:After 触发器的复杂性和执行频率可能会对数据库性能产生影响。因此,在创建 After 触发器时,建议遵循最佳实践,并进行性能测试和调优以确保系统性能不受负面影响。

1.2 DDL 触发器

DDL 触发器(Data Definition Language Trigger):这种触发器与数据库对象的定义操作(创建、修改、删除)相关。当执行与数据库对象相关的 DDL 语句时(如CREATE、ALTER、DROP),系统会自动调用执行与该对象相关的 DDL 触发器。DDL 触发器通常用于在对象定义发生变化时执行某些操作,如记录变更历史、限制数据库架构更改等。

触发器与普通的存储过程的区别在于触发器是与特定表相关联的,它们在特定的数据操作事件发生时自动触发执行,而不需要显式地调用。触发器提供了一种在数据库层面上实现业务逻辑和数据约束的机制,可以在数据操作发生时进行验证和处理。

1.3 INSERTED 表 与 DELETED 表

触发器有两个特殊的表:插入表(instered表)和删除表(deleted表)。这两张是逻辑表也是虚表。有系统在内存中创建者两张表,不会存储在数据库中。而且两张表的都是只读的,只能读取数据而不能修改数据。这两张表的结果总是与被改触发器应用的表的结构相同。当触发器完成工作后,这两张表就会被删除。Inserted表的数据是插入或是修改后的数据,而 deleted 表的数据是更新前的或是删除的数据。

  • DELETED 表存放由于执行 delete 或 update 语句(存放更新前的记录)而要从表中删除的所有行。
  • INSERTED 表存放由于执行 insert 或 update 语句(存放更新后的记录)而要向表中插入的所有行。

注意:触发器本身就是一个事务,所以在触发器里面可以对修改数据进行一些特殊的检查。如果不满足可以利用事务回滚,撤销操作。

对表的操作INSERTED 逻辑表DELETED 逻辑表
增加记录(insert)存放增加的记录
删除记录(delete)存放被删除的记录
修改记录(update)存放更新后的记录存放更新前的记录

二、T-SQL语句创建触发器各示例

2.1 基本通用语法

语法结构如下:

create trigger trigger_name
  on {table_name | view_name}
  {for | After | Instead of }
  [ insert, update,delete ]
  as
  sql_statement

2.2 创建 insert 类型触发器

insert 触发器,会在 inserted 表中添加一条刚插入的记录。

--创建insert插入类型触发器
if (object_id('tgr_classes_insert', 'tr') is not null)
    drop trigger tgr_classes_insert
go
create trigger tgr_classes_insert
on classes
    for insert --插入触发
as
    --定义变量
    declare @id int, @name varchar(20), @temp int;
    --在inserted表中查询已经插入记录信息
    select @id = id, @name = name from inserted;
    set @name = @name + convert(varchar, @id);
    set @temp = @id / 2;    
    insert into student values(@name, 18 + @id, @temp, @id);
    print '添加学生成功!';
go
--插入数据
insert into classes values('5班', getDate());
--查询数据
select * from classes;
select * from student order by id;

2.3 创建 delete 类型触发器

delete触发器会在删除数据的时候,将刚才删除的数据保存在deleted表中。

--delete删除类型触发器
if (object_id('tgr_classes_delete', 'TR') is not null)
    drop trigger tgr_classes_delete
go
create trigger tgr_classes_delete
on classes
    for delete --删除触发
as
    print '备份数据中……';    
    if (object_id('classesBackup', 'U') is not null)
        --存在classesBackup,直接插入数据
        insert into classesBackup select name, createDate from deleted;
    else
        --不存在classesBackup创建再插入
        select * into classesBackup from deleted;
    print '备份数据成功!';
go
--
--不显示影响行数
--set nocount on;
delete classes where name = '5班';
--查询数据
select * from classes;
select * from classesBackup;

2.4 创建 update 类型触发器

update 触发器会在更新数据后,将更新前的数据保存在 deleted 表中,更新后的数据保存在 inserted 表中。

--update更新类型触发器
if (object_id('tgr_classes_update', 'TR') is not null)
    drop trigger tgr_classes_update
go
create trigger tgr_classes_update
on classes
    for update
as
    declare @oldName varchar(20), @newName varchar(20);
    --更新前的数据
    select @oldName = name from deleted;
    if (exists (select * from student where name like '%'+ @oldName + '%'))
        begin
            --更新后的数据
            select @newName = name from inserted;
            update student set name = replace(name, @oldName, @newName) where name like '%'+ @oldName + '%';
            print '级联修改数据成功!';
        end
    else
        print '无需修改student表!';
go
--查询数据
select * from student order by id;
select * from classes;
update classes set name = '五班' where name = '5班';

2.5 update 更新列级触发器

更新列级触发器可以用update是否判断更新列记录;

if (object_id('tgr_classes_update_column', 'TR') is not null)
    drop trigger tgr_classes_update_column
go
create trigger tgr_classes_update_column
on classes
    for update
as
    --列级触发器:是否更新了班级创建时间
    if (update(createDate))
    begin
        raisError('系统提示:班级创建时间不能修改!', 16, 11);
        rollback tran;
    end
go
--测试
select * from student order by id;
select * from classes;
update classes set createDate = getDate() where id = 3;
update classes set name = '四班' where id = 7;

2.6 创建 instead of 触发器

if (object_id('tgr_classes_inteadOf', 'TR') is not null)
    drop trigger tgr_classes_inteadOf
go
create trigger tgr_classes_inteadOf
on classes
    instead of delete/*, update, insert*/
as
    declare @id int, @name varchar(20);
    --查询被删除的信息,病赋值
    select @id = id, @name = name from deleted;
    print 'id: ' + convert(varchar, @id) + ', name: ' + @name;
    --先删除student的信息
    delete student where cid = @id;
    --再删除classes的信息
    delete classes where id = @id;
    print '删除[ id: ' + convert(varchar, @id) + ', name: ' + @name + ' ] 的信息成功!';
go
--test
select * from student order by id;
select * from classes;
delete classes where id = 7;

三、修改触发器

alter trigger tgr_message
on student
after delete
as raisError('tgr_message 触发器被触发', 18, 12);
go
--test
delete from student where name = 'xiaoming';

四、启用、禁用触发器

--禁用触发器
disable trigger tgr_message on student;
--启用触发器
enable trigger tgr_message on student;

五、查询创建的触发器信息

--查询已存在的触发器
select * from sys.triggers;
select * from sys.objects where type = 'TR';

--查看触发器触发事件
select te.* from sys.trigger_events te join sys.triggers t
on t.object_id = te.object_id
where t.parent_class = 0 and t.name = 'tgr_valid_data';

--查看创建触发器语句
exec sp_helptext 'tgr_message';

六、结合触发器操作日志示例

下面进行一个举例,创建一个名为 log 的表和一个名为 tgr_student_log 的触发器。触发器用于在对 student 表进行插入、更新、删除操作时记录对应的动作到 log 表中。根据插入和删除操作是否存在,触发器分别插入不同的动作类型到 log 表中。最后,通过插入、更新、删除 student 表的操作进行测试,并查看 log 表和 student 表的结果。

if (object_id('log', 'U') is not null)
    drop table log
go
create table log(
    id int identity(1, 1) primary key,
    action varchar(20),
    createDate datetime default getDate()
)
go
if (exists (select * from sys.objects where name = 'tgr_student_log'))
    drop trigger tgr_student_log
go
create trigger tgr_student_log
on student
after insert, update, delete
as
    if ((exists (select 1 from inserted)) and (exists (select 1 from deleted)))
    begin
        insert into log(action) values('updated');
    end
    else if (exists (select 1 from inserted) and not exists (select 1 from deleted))
    begin
        insert into log(action) values('inserted');
    end
    else if (not exists (select 1 from inserted) and exists (select 1 from deleted))
    begin
        insert into log(action) values('deleted');
    end
go
--test
insert into student values('xiaoming', 23, 6, 7);
update student set sex = 0 where name = 'xiaoming';
delete student where name = 'xiaoming';
select * from log;
select * from student order by id;
	[ 本文作者 ]   bluetata
	[ 原文链接 ]   https://bluetata.blog.csdn.net/article/details/131093456
	[ 最后更新 ]   06/07/2023 19:58
	[ 版权声明 ]   如果您在非 CSDN 网站内看到这一行,
	说明网络爬虫可能在本人还没有完整发布的时候就抓走了我的文章,
	可能导致内容不完整,请去上述的原文链接查看原文。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/622543.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

WIN10:Cognos10.2_x32安装运行问题总结(二)

一、Cognos安装Congfiguration启动服务异常解决方法 1、用管理员权限启动IBM Cognos Configuration、关闭本机安全软件、关闭防火墙; 2、不卸载本机JDK不删除本机JDK环境,指定c10\bin目录,cogconfig -java:local 命令CMD启动Cognos。 测试…

JavaSE07_static和final

JavaSE-07 【static-final】 第一章 static关键字 1.1 静态static关键字概述 static关键字 一个类的不同对象有些共享的数据,这样我们就可以使用static来修饰 。一旦使用了static关键字,那么这样的内容不再属于对象,而是属于类的&#xff…

通过python封装接口seller_nick获取阿里巴巴店铺所有商品数据,阿里巴巴店铺所有商品数据接口,阿里巴巴API接口

今天给大家分享一个大体的思路和代码示例。请确保采集过程遵循相关网站的使用规则和政策。 使用阿里巴巴开放平台提供的API接口文档,找到seller_nick接口的具体参数及请求方式。使用ython中的requests库发送请求,获取接口返回的数据。对返回的数据进行处…

深度解读新档案法①:企业如何提高档案管理规范性?

上一次,我们列举了关于档案管理需要遵守的法律法规清单,想要了解详情的,可点击添加链接描述。关于如何依法进行档案管理,许多朋友仍有疑问。为此,我们开设了一个专栏,结合企业实践案例,详细解读…

7.5寸_Wifi标签功耗电流计算-单面屏幕-1.8V的Flash测试

7.5寸黑白屏幕Wifi标签不工作的时候电流的消耗主要来自两部分: 待机时,醒来监听时。经过测量,7.5寸黑白屏幕Wifi标签的待机电流是7.04uA , 醒来监听的峰值电流是80.1 mA。7.5寸黑白屏幕Wifi标签系统设定每间隔1秒醒来一次监听是否有唤醒数据…

PPT制作能力在职场中真的重要吗?一定要美术功底才能学好PPT吗?

前言 年轻时手握技术实力这把剑,我觉得闯荡江湖到处可以“仗义出剑”。实际做到一定级别后特别是我做到了架构师后发觉当不具备PPT能力后,我这条路反而越走越坚难了。 2009年时因为工作关系被企业推送了TOGAF培训,培训后以优秀的成绩拿到了…

vite+Vue3 createWebHistory 路由前缀配置

vue 作为前端最火的前端框架之一,大家在使用的过程中必不可少需要使用router(路由)。 路由有hash 和 history 两种,本期主要针对history模式下 路由前缀 使用和配置,进行简单介绍。 env变量 在根目录下,创建开发、测试、灰度、…

几种常见的 Python 数据结构

摘要:本文主要为大家讲解在 Python 开发中常见的几种数据结构。 数据结构和序列 元组 元组是一个固定长度,不可改变的 Python 序列对象。创建元组的最简单方式,是用逗号分隔一列值: In [1]: tup 4, 5, 6 当用复杂的表达式定义…

开放网关架构演进

淘宝开放平台是阿里与外部生态互联互通的重要开放途径,通过开放的产品技术把阿里经济体一系列基础服务,像水、电、煤一样输送给我们的商家、开发者、社区媒体以及其他合作伙伴,推动行业的定制、创新、进化, 并最终促成新商业文明生态圈。 开放…

最新ChatGPT网站源码+支持GPT4.0+Prompt设定+ai绘画+GPT3.5 API绘画+语音识别输入!

前言 程序已支持ChatGPT4.0、Midjourney绘画、GPT3.5 绘画、语音识别输入、文章资讯发布功能、用户每日签到功能一键更新版本。支持手机电脑不同布局页面自适应。 1、提问:程序已经支持GPT3.5、GPT4.0接口 2、支持三种Ai绘画模型(Midjourney模型、GPT3.5…

苹果Vision Pro物料成本曝光,3499美金售价原来是这样来的

苹果首款头显Vision Pro售价3499美元,近期网上流出了Vision Pro的物料清单(BOM),清单显示,Vision Pro物料成本约1509美元。 其中,二片内屏占700美元,为成本最高的零组件,由索尼供应…

谁说不可兼得,用scrcpy实现手机免流播放bilibili投屏电脑

目前的手机大额流量卡都是支持设备免流的,但是如何将这个流量用在其他设备,就相当麻烦。于是我查找了些相关Android投屏技术资料,发现了一个简单的USB投屏工具——scrcpy。 安装说明 Office:https://github.com/Genymobile/scrcp…

【JVM】JVM收集器CMS与G1区别

文章目录 区别一: 使用范围不一样区别二: STW的时间区别三: 垃圾碎片区别四: 垃圾回收的过程不一样CMS回收垃圾的4个阶段CMS的总结和优缺点G1回收器的特点大对象的处理G1回收垃圾的4个阶段什么情况下应该考虑使用G1G1设置参数 CMS…

Spark运行原理

1.BlockManager BlockManager是Spark的分布式存储系统 主从结构:BlockManagerMaster/BlockManager(Slave) BlockManagerMaster 在Driver端启动 :负责接受Executor上的BlockManager的注册 管理BlockManager的元数据信息 BlockMana…

浅谈养老所消防安全存在的问题与解决方案 安科瑞 许敏

1养老场所消防安全存在的问题 1.1筑物耐火等级低 养老场所新建的建筑较少,尤其是对于民办养老场所来说,大部分都是以原有的民房、工厂厂房、废弃学校等修建起来,通过简单的改造,而后演变成养老基地。这些建筑整体耐火等级不够&am…

海康工业相机WPF打开方式

1.通过nuget安装Gige和imageLibrary GitHub - Touseefelahi/GigeVision: Simple GigeVision implementation, GVSP, GVCP protocol implemented 2.窗体布局xaml代码 <Window x:Class"WpfApp1.MainWindow"xmlns"http://schemas.microsoft.com/winfx/2006/x…

面向对象学习笔记

常见代码块应用 局部代码块 在方法中出现&#xff0c;限定变量的生命周期构造代码块&#xff08;初始化块&#xff09; 在类中方法外出现&#xff0c;多个构造方法方法中相同的代码存放到一起&#xff0c;每次调用构造都执行&#xff0c;在调用构造方法前执行静态代码块 在类中…

达摩院+华为 | NLP博士的春招历程

作者 | luan2006 整理 | NewBeeNLP 面试锦囊之面经分享系列&#xff0c;持续更新中 赶紧后台回复"面试"加入讨论组交流吧 写在前面 本博均985&#xff08;不同校&#xff09;、现北京Top2计算机类博士生、方向为人工智能-自然语言处理。 个人优势&#xff1a;绩点年…

PeekingDuck

介绍 计算机视觉项目可能会非常令人望而生畏&#xff0c;涉及到各种工具和包&#xff0c;如OpenCV、TensorFlow和PyTorch等等。不仅需要熟悉所涉及的工具和API&#xff0c;还需要正确组合各个包&#xff0c;以使整个计算机视觉流水线正常工作。 例如&#xff0c;OpenCV以[H&…

玩转百问网东山Pi壹号-SSD202 Linux开发板(一)

点击上方“嵌入式应用研究院”&#xff0c;选择“置顶/星标公众号” 干货福利&#xff0c;第一时间送达&#xff01; 来源 | 嵌入式应用研究院 整理&排版 | 嵌入式应用研究院 很早之前就做过SSD20x相关的平台&#xff0c;刚好手里有一块东山Pi壹号-开发板&#xff0c;于是我…