SQL Server的CDC指的是"变更数据捕获"(Change Data Capture)。这是SQL Server数据库提供的一项功能,能够跟踪并记录对数据库表中数据所做的更改。这些更改包括插入、更新和删除操作。CDC可以捕获这些变更的详细信息,并使这些信息可用于进一步的处理,例如数据同步、数据仓库的更新或其他业务需求。
做了一个基础服务,定时扫cdc表的数据然后再数据同步到es上。运行一段时间后,发现es上的数据总是丢失变化。我想一个晚上,仍不知道自己错哪了。究竟是定时扫cdc的表拉取数据漏扫数据,还是数据处理出错呢,还是es同步数据出错呢?无奈日志打印内容太少了,只能增加日志重新部署生产观察了。
分析了一个晚上,终于找到原因了。
定时扫cdc的sql是类似这样
select * from table_CT with(nolock) where id>=@param order by id
扫到数据,处理完数据后把@param替换成上次扫的最大id,接着继续扫。
查询日志发现这条sql会漏扫数据。已知id是自增的,看着是没什么问题的。
with(nolock)是避免锁表,允许脏读(cdc表没有回滚的操作),第一眼看上去没什么问题。最读提前读到未提交的id,也没什么问题。
再细想,有问题,涉及到cdc如何捕获数据变更写入到变更表中。
把with(nolock)去掉,问题得到解决。
一看时间是凌晨4点钟,赶紧打电话给同事告诉他们这个好消息。