一例數據同步異常問題分析

来源:https://www.cnblogs.com/CtripDBA/archive/2019/04/23/10755110.html
-Advertisement-
Play Games

【問題描述】 開發反饋,有一個SQL Server數據同步的作業,從Table1 拉取數據,主鍵是ID, 每次拉取批次數據的SQL語句是 select top (15) from Table1(NOLOCK) where ID ?,?代表的是上次同步批次中最後一個ID號。 某一次拉取到的數據為 ID ...


【問題描述】

開發反饋,有一個SQL Server數據同步的作業,從Table1 拉取數據,主鍵是ID, 每次拉取批次數據的SQL語句是 select top (15) * from Table1(NOLOCK) where ID > ?,?代表的是上次同步批次中最後一個ID號。
某一次拉取到的數據為 ID: 8101102121,8101103081 兩條數據。查表發現,這兩條記錄中間,還有一條記錄,即ID=8101102855,這條記錄沒有被拉取到。十分困惑,為什麼會少拉一條記錄,是否拉取的時候,該條記錄沒有被創建?

Createtime ID Column3 Column4
  2019-04-11 14:17:14.843     8101102121           已處理
  2019-04-11 14:17:17.190     8101102855          已處理
  2019-04-11 14:17:20.237     8101103081          已處理

【問題分析】

剛開始看這個問題,也覺得非常奇怪。這個查詢語句中規中矩,從應用日誌來看,兩個ID之間的8101102855 確實是沒有被拉取到。
為進一步定位問題,我們分析具體的查詢語句。由於我們的伺服器開啟了XEvent Trace,我們定位到,當時在資料庫伺服器端真正執行的語句如下,比開發反饋的更多一些條件:

select top 15 id from table1 with (nolock) where id > 8101101700 and Column4 not like '%(特殊需求)%' and createtime > '2018-07-19 00:00:00.000' order by id asc

我們用這個查詢,在當前時間(2019-04-12 17:23:00)資料庫上進行查詢,確實能返回三條記錄。

通過資料庫明細記錄, 該語句在資料庫上的執行時間是:2019-04-11 14:17:21。對於ID=8101102855的記錄,其插入的時間是2019-04-11 14:17:17.190,比我們的查詢時間早4秒,按道理應該是能查出來的。另外,在2019-04-11 14:17:21的時候,ID=8101102855 的記錄正在被更新。難道是我們的查詢帶了NOLOCK,所以當前正在被更新的記錄跳過了?

根據我們的理解,NOLOCK相當於Read uncommitted, 是會讀取其他事務“修改後未提交的“數據。也就是說,是能夠讀出主鍵ID的。

【問題重現】

為了能夠更好的分析問題,我們定點還原資料庫,還原到2019-04-11 14:17:20的時候,也就是比應用的查詢時間早1秒鐘。其數據記錄如下。在這個點上,ID=8101103081 還沒有插入進來,但ID=8101102855,也就是我們關註的ID,數據已經進來了。

Createtime ID Column3 Column4
  2019-04-11 14:17:14.843     8101102121           已處理
  2019-04-11 14:17:17.190     8101102855          

針對上面的數據,我們執行查詢:

select top 15 id from table1 with (nolock) where id > 8101101700 and Column4 not like '%(特殊需求)%' and createtime > '2018-07-19 00:00:00.000' order by id asc

奇怪的發現,這個查詢只返回ID=8101102121,ID=8101102855沒有返回。經過簡單的調試,我們很快發現,是由於這個查詢條件所致:

Column4 not like '%(特殊需求)%'

ID=8101102855記錄在剛插入的時候,其Column4的值為NULL,從語義上來講,確實是不包含"(特殊需求)"這個字元串的。但在SQL處理上,卻和我們理解的不一樣。

SQL除了IS NULL和NOT NULL以外,只要出現NULL,值結果為FALSE。簡單來說,對於查詢SELECT * from table where name != ‘test’,只要name值是NULL,無論用name=’test’還是name != 'test', 都不能返回這一行。如果要返回的話,需要加IS NULL判斷:

SELECT * from table where name != ‘test’ or name IS NULL

至此,問題真相大白,解決方案也就很簡單:調整查詢語句,加一個IS NULL判斷即可。

select top 15 id from table1 with (nolock) where id > 8101101700 and (Column4 not like '%(特殊需求)%' or Column4 IS NULL) and createtime > '2018-07-19 00:00:00.000' order by id asc

【結論】

資料庫在碰到NULL的處理時候,要小心,查詢的判斷條件並不是很明顯,要註意IS NULL的情況。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1、通過yum安裝: yum install redis 2、設置redis.conf中daemonize為yes。設置密碼: requirepass 3、安裝完後的啟動腳本是完善的,/etc/init.d/redis,不需要再另外操作。 查看啟動列表: chkconfig --list 安裝完可以 ...
  • --語 句 功 能--數據操作SELECT --從資料庫表中檢索數據行和列INSERT --向資料庫表添加新數據行DELETE --從資料庫表中刪除數據行UPDATE --更新資料庫表中的數據 --數據定義 CREATE TABLE --創建一個資料庫表DROP TABLE --從資料庫中刪除表 A ...
  • 由於資料庫存的是整個字元串組到一起了,C#代碼是這個樣子的。 在sqlserver中存儲的實際值是:20190416124941。那麼直接轉換? 所以在sqlserver中查詢的時候我們要進行轉化,因為在mssql中進行轉換需要是有標準的 例如/ : 等符號。那麼我們就進行截取吧。 最後就完事了。 ...
  • 加鎖的主要目的是為了防止併發操作時導致的數據不一致等問題,鎖分為共用鎖(S)、更新鎖(U)、排他鎖(X),共用鎖與更新只是單向相容?傳說中的單相思? 事務 事務能保證數據操作的原子性,要麼內部操作都提交,要麼都回退。事務內部某個地方出錯時,可以回滾前面的操作,比如更新、刪除等。 共用鎖 共用鎖允許並 ...
  • 在資料庫的鎖機制中介紹過,資料庫管理系統(DBMS)中的併發控制的任務是確保在多個事務同時存取資料庫中同一數據時不破壞事務的隔離性和統一性以及資料庫的統一性。 樂觀併發控制(樂觀鎖)和悲觀併發控制(悲觀鎖)是併發控制主要採用的技術手段。 無論是悲觀鎖還是樂觀鎖,都是人們定義出來的概念,可以認為是一種 ...
  • 一、統計語句 1、--統計當前【>當天00點以後的數據】 SELECT * FROM 表 WHERE CONVERT(Nvarchar, dateandtime, 111) = CONVERT(Nvarchar, GETDATE(), 111) ORDER BY dateandtime DESC 2 ...
  • 安裝redis-5.0.4 修改配置文件 創建命令鏈接 設置redis開機自啟動 啟動redis服務 測試連接 安裝過程可能出現的問題 1.CentOS預設沒有安裝gcc,這會導致我們無法make成功 2.make時報如下錯誤: zmalloc.h:50:31: fatal error: jemal ...
  • [20190423]那個更快的疑問3.txt--//前一陣子,做了11g在單表單條記錄唯一索引掃描的測試,摘要如下:--//參考鏈接:http://blog.itpub.net/267265/viewspace-2636321/http://blog.itpub.net/267265/viewspa ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...