利用SQL Server触发器实现表的历史修改痕迹记录

 更新时间:2020年02月01日 16:21:23   投稿:mdxy-dxy  
在很多应用程序开发中,需要记录某些数据表的历史记录或修改痕迹,以便日后出现数据错误时进行数据排查。这种业务需求,我们可以通过数据库的触发器来轻松实现历史记录功能

在很多应用程序开发中,需要记录某些数据表的历史记录或修改痕迹,以便日后出现数据错误时进行数据排查。这种业务需求,我们可以通过数据库的触发器来轻松实现历史记录功能。

本文以SQL Server 2005数据库中的触发器为例(因为手中的项目用的就是这个数据库)

先简单描述一下SQL Server触发器。

SQL Server触发器的inserted和deleted

SQL Server为每个触发器都创建了两个专用虚拟表:inserted表和deleted表。这两个表由系统来维护,他们存在于内存中,而不是在数据库中。这两个表的结构总是与被该触发器作用的表结构相同。触发器执行完成后,与该触发器相关的两个表会被删除(即在内存中销毁)。

inserted表存放由执行insert或update语句而要想飙中插入的所有行;即:插入后或更新后的值。
deleted表存放由delete或update语句而要从表中删除的所有行;即:删除或更新钱的值。

SQL操作 inserted表 deleted表
增加(insert)记录 存放新增的记录 [不可用]
修改(update)记录 存放更新后的记录 存放更新前的记录
删除(delete)记录 [不可用] 存放被删除的记录

SQL Server触发器的instead of和after

SQL Server提供了两种触发器:instead of和after触发器。这两种触发器的区别在于他们被激活的时机不同:

  • instead of触发器用于替代引用触发器执行的sql语句。除表之外,instead of触发器也可以用于视图,用来扩展视图可以支持更新操作。
  • after触发器在一个inserted、update或delete语句之后执行,进行约束检查等动作都在after触发器被激活之前发生。after触发器只能用于数据表中。

说(复制)了这么多,是因为我们要实现的功能需要用到inserted虚拟表、deleted虚拟表和after触发器。

实现方法

通过一个示例来演示具体的实现方法。

假设当前有一个表:产品表(product),字段为“产品名(name)”、“产品描述(description)”、“单价(unit_cost)”和“生成日期(pub_time)”。

CREATE TABLE product(name VARCHAR(50),description VARCHAR(200),unit_cost MONEY,pub_time DATETIME)
GO

现在我们”上帝”的需求是:需要记录product表发生数据变化(增、删、改)时,记录每次操作改动情况。

1.创建日志表

需要创建一个产品日志表(product_log)用来将记录每次数据改动情况,我这里直接在原数据表的结构上增加两个字段(在实际开发环境中,大家可以根据需求来设置日志表的表结构),分别为sqlcomm和exectime;代码如下:

CREATE TABLE product_log(name VARCHAR(50),description VARCHAR(200),unit_cost MONEY,pub_time DATETIME,sqlcomm varchar(10),exectime datetime)
GO

新增的两个字段sqlcomm和exectime分别记录执行命令(insert、update和delete)和执行时间

2.增加触发器

在产品表增加触发器,其目的是为了记录表数据发生改变时记录到product_log中。

针对插入(insert)操作,增加名为tr_product_i的触发器:

CREATE TRIGGER tr_product_i
ON product
AFTER INSERT
AS
if @@rowcount = 0 --为了避免占用资源,当影响行数为0时,结束触发器
 return
insert into product_log (name,description,unit_cost,pub_time,sqlcomm,exectime)
 select name,description,unit_cost,pub_time,'insert',getdate() from inserted
GO

针对更新(update)操作,增加名为tr_product_u的触发器:

CREATE TRIGGER tr_product_u
ON product
AFTER UPDATE
AS
if @@rowcount = 0 --为了避免占用资源,当影响行数为0时,结束触发器
 return
/*更新前*/
insert into product_log (name,description,unit_cost,pub_time,sqlcomm,exectime)
 select name,description,unit_cost,pub_time,'update',getdate() from deleted
/*更新后*/
insert into product_log (name,description,unit_cost,pub_time,sqlcomm,exectime)
 select name,description,unit_cost,pub_time,'update',getdate() from inserted
GO

针对删除(delete)操作,增加名为tr_product_d的触发器:

CREATE TRIGGER tr_product_d
ON product
AFTER DELETE
AS
if @@rowcount = 0 --为了避免占用资源,当影响行数为0时,结束触发器
 return
insert into product_log (name,description,unit_cost,pub_time,sqlcomm,exectime)
 select name,description,unit_cost,pub_time,'delete',getdate() from deleted
GO

3.测试触发器

插入(insert)测试

INSERT INTO product(name,description,unit_cost,pub_time)
 VALUES('逗比','这是一个逗比的测试数据',200.5,'1990-11-18')
GO

SELECT * FROM product
SELECT * FROM product_log
GO

更新(update)测试

UPDATE product SET unit_cost=250.0 WHERE name='逗比'
GO

SELECT * FROM product
SELECT * FROM product_log
GO

删除(delete)测试

DELETE FROM product WHERE name='逗比'
GO

SELECT * FROM product
SELECT * FROM product_log
GO

好了这篇文章就介绍到这了,需要的朋友可以参考一下。

相关文章

  • SQL查询某列指定长度的字符串多余的用省略号来表示

    SQL查询某列指定长度的字符串多余的用省略号来表示

    有时候为了美观,只需要显示前面几个字符串,剩下的可以用省略号来表示,下面有个不错的示例,感兴趣的朋友可以参考下
    2013-11-11
  • SQL中的partition分区功能使用详解

    SQL中的partition分区功能使用详解

    本文介绍了SQL中的分区功能,包括使用ROW_NUMBER()窗口函数和PARTITION BY进行数据分组和排序,以及如何在创建表时进行物理分区以优化查询性能,感兴趣的朋友跟随小编一起看看吧
    2024-12-12
  • SQLSERVER服务手工启动 批处理文件

    SQLSERVER服务手工启动 批处理文件

    装完SQLSERVER2005 因为开机很慢 所以呢就把开机服务搞成手动的了,这样开机快了很多可是问题也来了每次要用SqlServer的时候要一个一个服务去开起来
    2011-03-03
  • SqlServer身份验证登录配置的实现步骤

    SqlServer身份验证登录配置的实现步骤

    SQL Server身份验证是一种用于验证用户身份的方法,本文主要介绍了SqlServer身份验证登录配置,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2024-04-04
  • SQL Server在AlwaysOn中使用内存表的“踩坑”记录

    SQL Server在AlwaysOn中使用内存表的“踩坑”记录

    这篇文章主要给大家介绍了关于SQL Server在AlwaysOn中使用内存表的一些"踩坑"记录,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习下吧。
    2017-09-09
  • 优化 SQL Server 索引的小技巧

    优化 SQL Server 索引的小技巧

    SQL Server中有几个可以让你检测、调整和优化SQL Server性能的工具
    2012-08-08
  • SQL Server 分页查询通用存储过程(只做分页查询用)

    SQL Server 分页查询通用存储过程(只做分页查询用)

    这段存储过程写得SQL代码很不错,在这个基础上,按照我的习惯以及思维方式,调整了代码,只做分页查询用
    2014-07-07
  • mssql 建立索引

    mssql 建立索引

    假设你想找到本书中的某一个句子。你可以一页一页地逐页搜索,但这会花很多时间。而通过使用本书的索引,你可以很快地找到你要搜索的主题。
    2009-04-04
  • 安装SQL Server2019详细教程(推荐!)

    安装SQL Server2019详细教程(推荐!)

    SQL Server数据库是Microsoft开发设计的一个关系数据库智能管理系统(RDBMS),现在是全世界主流数据库之一,下面这篇文章主要给大家介绍了关于安装SQL Server2019详细教程,需要的朋友可以参考下
    2022-11-11
  • 使用 SQL 服务器时,

    使用 SQL 服务器时,"评估期已过期"错误消息(解决方法)

    这篇文章主要介绍了使用 SQL 服务器时,"评估期已过期"错误消息,本文分步骤给大家分享解决方法,需要的朋友可以参考下
    2019-12-12

最新评论