捕获Insert触发器失败记录
1、背景
环境:发布服务器A Windows2008+SQL2008,分发服务器B Windows2008+SQL2008,订阅服务器C Windows2008+SQL2012
发布服务器A上的用户信息表member,通过事务复制,推送到订阅器C上的用户信息表member。订阅服务器C上存在一张用户账户表,之前的机制是通过在用户信息表上的insert触发器往用户账户表初始化数据。
问题:发布服务器上的用户信息表的某一字段与订阅服务器上的用户信息表的某一字段内容不一致,近一个月出现了两次,运营反馈这个问题已经持续很长一段时间,希望我们能彻底解决,而不是每次用户反馈了再查再同步!
处理:主库导一份用户数据到订阅数据库上,对比不一致的字段有多少记录,无意中发现订阅的数据量少于发布的记录数。问题重心由字段不一致转移到数量不一致,是什么原因导致记录条数都不一样呢?启动复制监视器,查看分发到订阅的历史记录,发现下面的错误信息,查看这些时间点主库有记录插入,但订阅上找不到对应记录。
此时基本可以想到是插入过程中出了问题,如果有分发库上的权限完全可以查出具体的错误信息。现在假设数据已经传到订阅数据库,由于某种原因,导致数据没能插入成功。
2、模拟触发器
模拟生产环境下通过触发器将用户表中的数据插入到用户账户信息表。
create table test1 (id int ,name varchar(20),date datetime default getdate())
create table test2 (id int ,name varchar(20),date datetime default getdate())
create unique nonclustered index [uniqueindex] on [dbo].[test2] ([id] asc)
create trigger tr_test1
on test1
for insert
as
begin
insert into test2(id,name,date)
select * from inserted
end
--连续插入两次
insert into test1 select 1,'abs',getdate()
test2有唯一索引,当往test1中插入相同记录时,会失败(test1、test2插入失败)。
如何将失败的记录获取出来,尝试在触发器用try-catch,并把失败的记录插入到另一张表test3中
create table test3 (id int ,name varchar(20),date datetime default getdate())
--修改触发器
alter trigger tr_test1 on test1
for insert
as
begin
begin try
insert into test2(id,name,date )
select * from inserted
end try
begin catch
insert into test3(id,name,date )
select * from inserted
end catch
end
--插入测试数据
insert into test1 select 1,'abs',getdate()
当插入相同记录时,同样报错,只是错误信息有所不同
群里请教别人,很快有群友回复
alter trigger tr_test1 on test1
for insert
as
begin
set xact_abort off
begin try
insert into test2(id,name,date )
select * from inserted
end try
begin catch
insert into test3(id,name,date )
select * from inserted
end catch
end
--插入测试数据
insert into test1 select 1,'ere',getdate()
重点是SET XACT_ABORT OFF,此时往Test1插入id已存在于Test2中的记录,就不会报之前的错误。查询三张表的记录如下:
至此已经明白订阅表为什么与发布表不一致。主库每天会按照一定规则从用户信息表删除僵尸用户,订阅表同步删除,但是用户账户表前期并没有删除对应记录。之后ID重用,往订阅端用户信息表写入数据的同时,由于触发器需写入到用户账户表,但是用户账户表在ID字段有唯一约束,insert失败!就出现发布与订阅不一致的情况。
解决方法:确保从用户信息表删除僵尸用户的同时,也从用户账户表删除记录。还可以适当修改触发器让已存在用户账户表的新记录写入到异常用户日志表,方便核对!
3、延伸
将表Test1重命名,然后再创建同结构的Test1,之后在Test1上创建一个Insert触发器。通过sp_helptext triggername查看,此时有两个触发器在Test1上。可实际用的应该是哪个?
--重命名test1表
sp_rename 'test1','test1_alias','object'
--重新创建test1数据表
create table test1 (id int ,name varchar(20),date datetime default getdate())
--重新创建test1上的触发器,为区分跟踪在里面增加一行注释
create trigger tr_test1_alias on test1
for insert
as
begin
set xact_abort off
begin try
insert into test2(id,name,date )--区分跟踪
select * from inserted
end try
begin catch
insert into test3(id,name,date )--区分跟踪
select * from inserted
end catch
end
此时用sp_helptext查看触发器的定义,发现两个都是在test1上
在sysobjects中看触发器的parent_object
--先开启跟踪,然后插入记录
insert into test1 select 2,'tst',getdate()
跟踪信息如下,图片中可以看出插入数据使用的是后面创建的触发器tr_test1_alias
此时查询三张表中的记录信息如下所示:
可以在test1_alais上插入数据,看其上的触发器能否使用
--先开启跟踪,然后插入记录
insert into test1_alias select 3,'uyu',getdate()


从跟踪信息和数据结果可以看出插入数据使用的是触发器tr_test1,虽然sp_helptext tr_test1返回是在test1上,但如果我们在对象资源管理器中,右击对应触发器然后修改或编辑触发器脚本到新窗口,可以看到其指向的是test1_alias。重命名表,与其相关的触发器的parent_object_id指向新表,触发器以对象资源管理器下看到的为准。
捕获Insert触发器失败记录的更多相关文章
- [转]连续创建多个Oracle触发器失败,单个创建才成功的解决方法
连续创建多个Oracle触发器失败,单个创建才成功的解决方法 1.当我连续执行创建多个触发器时,总是报编译通过,但存在警告或错误.如下: create or replace trigger t ...
- SQLServer之创建DML AFTER INSERT触发器
DML AFTER INSERT触发器创建原理 触发器触发时,系统自动在内存中创建deleted表或inserted表,内存中创建的表只读,不允许修改,触发器执行完成后,自动删除. insert触发器 ...
- 记一次SQL Server Insert触发器编写过程
实现功能:新增特定类型的新闻时,自动追加特定的背景图片. 第一版(错误信息:不能在 'inserted' 表和 'deleted' 表中使用 text.ntext 或 image 列),代码如下: - ...
- 【转】MySQL 当记录不存在时insert,当记录存在时update
MySQL当记录不存在时insert,当记录存在时更新:网上基本有三种解决方法 第一种: 示例一:insert多条记录 假设有一个主键为 client_id 的 clients 表,可以使用下面的语句 ...
- MySQL 当记录不存在时insert,当记录存在时update(ON DUPLICATE KEY UPDATE, REPLACE语句)
MySQL 当记录不存在时insert,当记录存在时更新 网上基本有三种解决方法. 第一种:示例一:insert多条记录 假设有一个主键为 client_id 的 clients 表,可以使用下面的语 ...
- MySQL 当记录不存在时insert,当记录存在时update
MySQL当记录不存在时insert,当记录存在时更新:网上基本有三种解决方法 第一种: 示例一:insert多条记录 假设有一个主键为 client_id 的 clients 表,可以使用下面的语句 ...
- MySQL 当记录不存在时insert,当记录存在时更新
网上基本有三种解决方法. 第一种: 示例一:insert多条记录 假设有一个主键为 client_id 的 clients 表,可以使用下面的语句: INSERT INTO clients (clie ...
- SpringMVC 捕获参数绑定失败时的异常
SpringMVC配置数据验证(JSR-303)中提到了用String类型的域来绑定Ajax中的非法类型的参数. 这样做的目的是一旦发生一种情况,后端可以返回一个自定类的返回值,而不是返回Spring ...
- 记一次Tomcat运行失败记录
记一次Tomcat运行失败记录 如图tomcat运行之后会出现这样的情况,在网上百度之后大部分都说的是web.xml或者其他配置文件的问题,但是根据网上修改了之后却还是老样子. 这里有比较好的网址可以 ...
随机推荐
- 为OpenResty增加ngx_pagespeed模块进行优化
1.下载ngx_pagespeed模块 wget https://github.com/pagespeed/ngx_pagespeed/archive/v1.8.31.4-beta.zip unzip ...
- IOS UI segmentedControl UISegmentedControl 常见属性和用法
UISegmentedControl中一些常见的属性和用法 //设置以图案作为分段的显示,仅需要图案的轮廓,这样颜色为分段的背景颜色 // NSArray *items = @[[UIImage ...
- cdoj 1334 郭大侠与Rabi-Ribi Label:贪心+数据结构
郭大侠与Rabi-Ribi Time Limit: 3000/1000MS (Java/Others) Memory Limit: 65535/65535KB (Java/Others) 最近 ...
- [Leetcode] Permutation Sequence
The set [1,2,3,…,n] contains a total of n! unique permutations. By listing and labeling all of the p ...
- JS实现屏蔽键盘操作
第一种:当页面初始加载的时候,屏蔽掉当前页面所有的键盘 $(document).ready(function () { document.body.onkeydown = function (even ...
- C#中的IComparable 和 IComparer 接口,实现列表中的对象比较和排序
借豆瓣某博主的话先对这两个接口进行一个解释: IComparable在要比较的对象的类中实现,可以比较该对象和另一个对象 IComparer在一个单独的类中实现,可以比较任意两个对象. 如果已经支持 ...
- 【BZOJ】1532: [POI2005]Kos-Dicing
题意 \(n\)个人\(m\)场比赛\((1 \le n \le 10000, 0 \le m \le 10000)\),给出每场比赛的两个选手,求赢得最多的人最少赢的场数. 分析 二分最多人赢的场数 ...
- poj 1995 裸快速幂
1. poj 1995 Raising Modulo Numbers 2.链接:http://poj.org/problem?id=1995 3.总结:今天七夕,来发水题纪念一下...入ACM这个坑 ...
- js之获取窗口大小和位置信息
除IE外的浏览器查看窗口大小和位置信息: //The overall size of the browser window on the desktop var windowWidth = windo ...
- PostgreSQL新手入门
自从MySQL被Oracle收购以后,PostgreSQL逐渐成为开源关系型数据库的首选. 本文介绍PostgreSQL的安装和基本用法,供初次使用者上手.以下内容基于Debian操作系统,其他操作系 ...