sqlserver触发器修改当前字段(利用SQL Server触发器实现表的历史修改痕迹记录)
sqlserver触发器修改当前字段
利用SQL Server触发器实现表的历史修改痕迹记录在很多应用程序开发中,需要记录某些数据表的历史记录或修改痕迹,以便日后出现数据错误时进行数据排查。这种业务需求,我们可以通过数据库的触发器来轻松实现历史记录功能。
本文以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
好了这篇文章就介绍到这了,需要的朋友可以参考一下。
- centos7.5安装mysql教程(CentOS8下MySQL 8.0安装部署的方法)
- thinkphp伪静态实例(thinkPHP+mysql+ajax实现的仿百度一下即时搜索效果详解)
- mysql触发器怎么添加(MySQL触发器的使用场景及方法实例)
- sqlserver列数据拆分(SQL Server基础之行数据转换为列数据)
- MySQL DATE_FORMAT()用法
- mysql8.0中如何导入数据(mysql8.0.20数据目录迁移的方法)
- sql解析器配置(sql字段解析器的实现示例)
- mysql显示所有数据库语句(MySQL数据库自动补全命令的三种方法)
- mariadb导入数据库命令(MySQL/MariaDB 如何实现数据透视表的示例代码)
- php提供哪些函数来避免sql的注入(PHP与SQL语句写一句话木马总结)
- mysqlworkbench怎么设置连接(详解MySQL Workbench使用教程)
- sql server 2016配置管理(SQL Server 2016 配置 SA 登录教程)
- sql server 高并发update 死锁(解密新型SQL Server无文件持久化恶意程序的问题)
- sql语言运算符主要内容(详解SQL EXISTS 运算符)
- mysql索引原理及使用(再有人问你MySQL索引原理,就把这篇文章甩给他!)
- 如何查看linq生成的sql
- 萌娃唱《白龙马》走红,那生动的小表情,网友直呼 简直是戏精(萌娃唱白龙马走红)
- 朱鹤松被不断认可,凤凰传奇玲花喊话岳云鹏,索要老朱演出门票(朱鹤松被不断认可)
- 元宵晚会槽点多,芒果台上来就假唱,岳云鹏不说相声改评书了(元宵晚会槽点多)
- 岳云鹏跟凤凰传奇谈心,说出了人生中最重要的三个人,这才成功(岳云鹏跟凤凰传奇谈心)
- 爱情可以当饭吃吗(爱情能当饭吃吗)
- Top 3 JSHS《运动与健康科学 英文 》跻身SCI体育学期刊世界前三(Top3JSHS运动与健康科学)
热门推荐
- Extjs msgTarget 提示位置
- css3粒子特效(利用CSS3实现毛玻璃效果示例源码)
- docker中搭建jmeter测试环境(借助Docker搭建JMeter+Grafana+Influxdb监控平台的详细教程)
- 导航栏制作步骤详细(导航栏的多样设置简单实例)
- react的动画实现(React实现动效弹窗组件)
- python获取excelsheet名称(python查询文件夹下excel的sheet名代码实例)
- docker 镜像id修改(Docker Alpine镜像时区问题完美解决方案)
- docker配置自己的环境(docker可视化工具Portainer部署并汉化的操作)
- docker创建容器使用教程(Docker容器连接实现步骤解析)
- 怎么调input搜索框边框颜色(使用placeholder属性设置input文本框的提示信息)
排行榜
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9