為什麼忘記commit也會造成select查詢的效能問題

發表於2016-09-13

今天遇到一個很有意思的問題,一個開發人員反饋在測試伺服器ORACLE資料庫執行的一條簡單SQL語句非常緩慢,他寫的一個SQL沒有返回任何資料,但是耗費了幾分鐘的時間。讓我檢查分析一下原因,分析解決過後,發現事情的真相有點讓人哭笑不得,但是也是非常有意思的。我們先簡單構造一下類似的案例,當然只是簡單模擬。

假設一個同事A,建立了一個表並初始化了資料(實際環境資料量較大,有1G多的資料),但是他忘記提交了。我們簡單模擬如下:


clip_image001

另外一個同事B對這個表做一些簡單查詢操作,但是他不知道同事A的沒有提交INSERT語句,如下所示,查詢時間用了大概5秒多(這個因為構造的資料量不是非常大的緣故。實際場景耗費了幾分鐘)


clip_image002

當時是在SQL Developer工具裡面分析SQL的執行計劃,並沒有注意到redo size非常大的情況。剛開始懷疑是統計資訊不準確導致,手工收集了一下該表的統計資訊,執行的時間和執行計劃依然如此,沒有任何變化。 如果我們使用SQL*Plus,檢視執行計劃,就會看到redo size異常大,你就會有所察覺(見後面分析)


因為ORACLE裡面的寫不阻塞讀,所以不可能是因為SQL阻塞的緣故,然後我想檢視這個表到底有多少記錄,結果亮瞎了我的眼睛,記錄數為0,但是空間用掉了852 個資料塊


clip_image003

於是我使用Tom大師的show_space指令碼檢查、確認該表的空間使用情況,如下所示,該表確實使用852個資料塊。


clip_image004

分析到這裡,那麼肯定是遇到了插入資料操作,卻沒有提交的緣故。用下面指令碼檢查發現一個會話ID為883的對這個表有一個ROW級排他鎖,而且會話還有一個事務排他鎖,那麼可以肯定這個會話執行了DML操作,但是沒有提交。


clip_image005

我們在會話裡面提交後,然後重新執行這個SQL,你會發現執行計劃裡面redo size為0,這是因為redo size表示DML生成的redo log的大小,其實從上面的執行計劃分析redo size異常,就應該瞭解到一個七七八八了,因為一個正常的SELECT查詢是不會在redo log裡面生成相關資訊的。那麼肯定是遇到了DML操作,但是沒有提交。

clip_image006

分析到這裡,我們已經知道事情的前因後果了,解決也很容易,找到那個會話的資訊,然後定位到哪個同事,讓其提交即可解決。但是,為什麼沒有提交與提交過後的差距那麼大呢?是什麼原因呢? 我們可以在這個案例,提交前與提交後跟蹤執行的SQL語句,如下所示。


提交前上面SQL生成的跟蹤檔案為scm2_ora_8444.trc,我們使用TKPROF格式化如下: tkprof scm2_ora_8444.trc out_uncommit.txt 如下所示

clip_image007

提交後,在另外一個會話執行上面的SQL,然後格式化跟蹤檔案如下所示:

clip_image008

我們發現提交前與提交後兩者的物理讀、一致性讀有較大差別(尤其是一致性讀相差3倍多)。這個主要是因為ORACLE的一致性讀需要構造cr塊,產生了大量的邏輯讀的緣故。相關理論與概念如下:

為什麼要一致性讀,為了保持資料的一致性。如果一個事務需要修改資料塊中資料,會先在回滾段中儲存一份修改前資料和SCN的資料塊,然後再更新Buffer Cache中的資料塊的資料及其SCN,並標識其為“髒”資料。

當其他程式讀取資料塊時,會先比較資料塊上的SCN和程式自己的SCN。如果資料塊上的SCN小於等於程式本身的SCN,則直接讀取資料塊上的資料;

如果資料塊上的SCN大於程式本身的SCN,則會從回滾段中找出修改前的資料塊讀取資料。通常,普通查詢都是一致性讀。

一致性讀什麼時候需要cr塊呢,那就是select語句在發現所查詢的時間點對應的scn,與資料塊當前所的scn不一致的時候。構造cr塊的時候,首先去data buffer中去找包含資料庫前映象的undo塊,如果有直接取出構建CR塊,這時候是邏輯讀,產生邏輯IO;但是data buffer將undo資訊寫出後,就沒有需要的undo資訊,就會去undo段找所需要的前映象的undo資訊,這時候從磁碟上讀出block到buffer中,這時候產生物理讀(物理IO)

相關文章