利用SQL Server触发器实现表的历史修改痕迹记录
作者:mdxy-dxy 发布时间:2024-01-19 18:06:42
在很多应用程序开发中,需要记录某些数据表的历史记录或修改痕迹,以便日后出现数据错误时进行数据排查。这种业务需求,我们可以通过数据库的触发器来轻松实现历史记录功能。
本文以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
好了这篇文章就介绍到这了,需要的朋友可以参考一下。
来源:https://wuzhuti.cn/sql-server-trigger-data-history


猜你喜欢
- 源码及注释:import pygamefrom sys import exitfrom random import randintimpor
- 1.open使用open打开文件后一定要记得调用文件对象的close()方法。比如可以用try/finally语句来确保最后能关闭文件。fi
- 前言Python本身已有顺序表(List、Tupple)的实现,所以这里从栈开始。什么是栈想象一摞被堆起来的书,这就是栈。这堆书的特点是,最
- function getBytesLength(str){ var re=/[\x00-\xf
- 解释器模式(Interpreter Pattern)是什么解释器模式是一种行为型模式,它定义了一种语言文法,并且定义了一个解释器,用来解释这
- Pycharm的下方工具栏中有两个窗口:Python Console和Terminal(如下图)其中,Python Console叫做Pyt
- PHP mysqli_select_db() 函数更改连接的默认数据库:删除数据库<?php // 假定数据库用户名:root,密码:
- file_get_contents的超时处理话说,从PHP5开始,file_get_content已经支持context了(手册上写着:5.
- 以下实例用于判断一个数字是否为奇数或偶数:# -*- coding: UTF-8 -*-# Filename : test.py# Pyth
- 使用python3+opencv3.3.1环境将视频流保存为本地视频文件,具体内容如下1、利用opencv中的VideoCapture类获取
- Python字典的基本用法创建字典:myDict1 = { '薛之谦':'我叫薛之谦', &nb
- 本文实例讲述了python使用pil生成图片验证码的方法。分享给大家供大家参考。具体实现方法如下:# -*- coding: utf-8 -
- 本文将介绍 5 种基于 Plotly 的可视化方法,你会发现,原来可视化不仅可用直方图和箱形图,还能做得如此动态好看甚至可交互。那么,Plo
- 用Python生成一个简单的密码本,一般是有数字、字母和符号组成,这里用到的思路主要是穷举法。通过使用pywifi 模块,根据密码本暴力破解
- 用Django开发网站的时候,前端页面内的文本框总是不能被设置为只读,找了一些资料发现可以在form class里面进行设置。方法一:首先在
- 在以前的一篇文章自动刷新 从BrowserSync开始中,我介绍了BrowserSync这样一个出色的开发工具。通过BrowserSync我
- VSCode卸载后进行重新安装,发现新安装的还有原来的一些配置,卸载的不彻底,有时候也容易出问题,可按照如下方法卸载干净:1.进入控制面板卸
- 修改数据库为多用户模式SQL语句修改数据位多用户模式如下:DECLARE @SQL VARCHAR(MAX); SET @SQL='
- 描述在linux中获取进程cmdline时遇到隐藏符号问题,如下:[root@vm010066016161 /root]#cat /proc
- selenium 安装与 chromedriver安装我们前文提到,Python脚本中使用了selenium库,而selenium又通过ch