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触发器失败记录的更多相关文章

  1. [转]连续创建多个Oracle触发器失败,单个创建才成功的解决方法

    连续创建多个Oracle触发器失败,单个创建才成功的解决方法   1.当我连续执行创建多个触发器时,总是报编译通过,但存在警告或错误.如下:   create or replace trigger t ...

  2. SQLServer之创建DML AFTER INSERT触发器

    DML AFTER INSERT触发器创建原理 触发器触发时,系统自动在内存中创建deleted表或inserted表,内存中创建的表只读,不允许修改,触发器执行完成后,自动删除. insert触发器 ...

  3. 记一次SQL Server Insert触发器编写过程

    实现功能:新增特定类型的新闻时,自动追加特定的背景图片. 第一版(错误信息:不能在 'inserted' 表和 'deleted' 表中使用 text.ntext 或 image 列),代码如下: - ...

  4. 【转】MySQL 当记录不存在时insert,当记录存在时update

    MySQL当记录不存在时insert,当记录存在时更新:网上基本有三种解决方法 第一种: 示例一:insert多条记录 假设有一个主键为 client_id 的 clients 表,可以使用下面的语句 ...

  5. MySQL 当记录不存在时insert,当记录存在时update(ON DUPLICATE KEY UPDATE, REPLACE语句)

    MySQL 当记录不存在时insert,当记录存在时更新 网上基本有三种解决方法. 第一种:示例一:insert多条记录 假设有一个主键为 client_id 的 clients 表,可以使用下面的语 ...

  6. MySQL 当记录不存在时insert,当记录存在时update

    MySQL当记录不存在时insert,当记录存在时更新:网上基本有三种解决方法 第一种: 示例一:insert多条记录 假设有一个主键为 client_id 的 clients 表,可以使用下面的语句 ...

  7. MySQL 当记录不存在时insert,当记录存在时更新

    网上基本有三种解决方法. 第一种: 示例一:insert多条记录 假设有一个主键为 client_id 的 clients 表,可以使用下面的语句: INSERT INTO clients (clie ...

  8. SpringMVC 捕获参数绑定失败时的异常

    SpringMVC配置数据验证(JSR-303)中提到了用String类型的域来绑定Ajax中的非法类型的参数. 这样做的目的是一旦发生一种情况,后端可以返回一个自定类的返回值,而不是返回Spring ...

  9. 记一次Tomcat运行失败记录

    记一次Tomcat运行失败记录 如图tomcat运行之后会出现这样的情况,在网上百度之后大部分都说的是web.xml或者其他配置文件的问题,但是根据网上修改了之后却还是老样子. 这里有比较好的网址可以 ...

随机推荐

  1. Button 设置适应不同版本 旋转以后大小相应的改变

    很多时候对于不同的版本,随设备的旋转以后,相应的Button的大小如果不做相应的改变,这很影响视图的美观和布局:下面是小编的个人看法  UIButton *button = [[UIBtton all ...

  2. 【POJ】2096 Collecting Bugs

    http://poj.org/problem?id=2096 题意:s个系统n种bug,每天找出一个bug,种类的概率是1/n,系统的概率是1/s.问:每个系统至少找出一个bug:每种类的bug都被找 ...

  3. 翻译-In-Stream Big Data Processing 流式大数据处理

    相当长一段时间以来,大数据社区已经普遍认识到了批量数据处理的不足.很多应用都对实时查询和流式处理产生了迫切需求.最近几年,在这个理念的推动下,催生出了一系列解决方案,Twitter Storm,Yah ...

  4. docker warning ipv4 forwarding is disabled. networking will not work

    # vi /etc/sysctl.conf 添加如下代码:     net.ipv4.ip_forward=1 重启network服务 # systemctl restart network   查看 ...

  5. 1019 JDBC链接数据库进行修删改查

    package com.liu.test01; import java.sql.Statement; import java.sql.Connection; import java.sql.Drive ...

  6. Java读取Execl表格数据

    在前面提到用java代码新建一个Execl 表格并添加数据到表格中, 这次写了一个读取Execl表格数据并添加导数据库中的案列 给定对方一个Execl模板表格,如果导入的Execl表格和预订的表格不相 ...

  7. html图片和文字的细节

    ul中的每一个li如果里面添加“一个图,一行字”, 这样图片会紧贴在左侧,而文字会居中,这两个元素不会紧贴着. 产生这种问题的原因我推测是:我图片设置了左浮动,但文字没有设置浮动,而一旦将文字设置为浮 ...

  8. jq图片切换特效

    首先引入js,内容如下: (function($){$.fn.slides=function(option){option=$.extend({},$.fn.slides.option,option) ...

  9. 状态压缩 DP

    D - Hie with the Pie Crawling in process... Crawling failed Time Limit:2000MS     Memory Limit:65536 ...

  10. Jquery dialog属性

    修改标题: $('#test').dialog("option","title", "测试").dialog('open'); 修改位置: ...