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. 使用 google gson 转换Timestamp或Date类型为JSON字符串.

    http://blog.csdn.net/z69183787/article/details/13016289 创建类型适配类: import java.lang.reflect.Type; impo ...

  2. 20161005 NOIP 模拟赛 T3 解题报告

    subset 3.1 题目描述 一开始你有一个空集,集合可以出现重复元素,然后有 Q 个操作 1. add s 在集合中加入数字 s. 2. del s 在集合中删除数字 s.保证 s 存在 3. c ...

  3. Java_解密ThreadLocal

    概述 相信读者在网上也看了很多关于ThreadLocal的资料,很多博客都这样说:ThreadLocal为解决多线程程序的并发问题提供了一种新的思路:ThreadLocal的目的是为了解决多线程访问资 ...

  4. 20145330第十周《Java学习笔记》

    20145330第十周<Java学习笔记> 网络编程 网络编程就是在两个或两个以上的设备(例如计算机)之间传输数据.程序员所作的事情就是把数据发送到指定的位置,或者接收到指定的数据,这个就 ...

  5. 如何将maven项目导入MyEclipse

    一.安装maven第一步:下载一个免安装版的apache-maven-3.0.3.zip解压后,配置环境变量 新建M2_HOME:   在path后面添加  %M2_HOME%\bin;   第二步: ...

  6. MVVM deep dive

    You can get a different instance each time by passing a different key to the GetInstance method. How ...

  7. 1071. Speech Patterns (25)

    People often have a preference among synonyms of the same word. For example, some may prefer "t ...

  8. 将普通工程转为mvn标准工程(main resources)

    It is sometimes required to change the default source folder working on the java project. One best e ...

  9. Android使用Fragment定义弹出数字键盘

    fragment主布局文件 <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmln ...

  10. Asp文件锁定脚本

    锁定指定文件 <% on error resume next server.ScriptTimeout= response.write "<form method=post> ...