ABAP面試題系列:寫一組會出現死鎖(Deadlock)的ABAP程式
我們在計算機作業系統這門專業課上,學過死鎖(Deadlock)的概念:兩個或兩個以上的程式(或執行緒)在執行過程中,由於競爭資源而造成的一種阻塞的現象,稱為死鎖。若無外力干預,這些處於死鎖狀態的程式將永遠處於互相等待的阻塞狀態中。
正好我兒子走到我電腦前看到文章標題,好奇地問我什麼是死鎖。我解釋道,“假設你和白妹妹(他的玩伴)手上都有一張奧特曼白金卡,你特別想要白妹妹手上那張白金卡,白妹妹也特別想要你手上那張白金卡。你們都想讓對方把他/她手上那張卡送給你們,但你們都捨不得把自己手上那張卡送出去。這就是死鎖。”
兒子又問,那這種情況咋辦。
我說,只有靠大人的介入。比如你老爸出馬,把你們手中兩張卡都沒收了,等我玩夠了再還給你們,這樣你們就不會死鎖了。
不過Jerry仔細端詳了這些一零後們喜歡玩的奧特曼卡片,發現毫無吸引我之處。我還是更喜歡欣賞和收藏這些反映祖國傳統文化的水滸傳人物卡片。
言歸正傳,在使用ABAP答這道面試題之前,我們先看看如何用Java編寫一個會出現死鎖的程式。
不到40行程式碼就完成任務。為了便於ABAP從業人員理解,沒有使用Java裡的Lambda表示式,否則程式碼可以更短。
該程式邏輯如下:
執行緒1持有資源1,試圖持有資源2; 執行緒2持有資源2,試圖持有資源1; 死鎖發生;
現在進入ABAP部分。
ABAP幫助文件提到,使用SELECT SINGLE FOR UPDATE讀取單條記錄時,會自動在資料庫層面為該條記錄設定一把鎖(Exclusive lock,有的中文文件翻譯成排他鎖)。如果上鎖操作會導致死鎖發生,則會丟擲異常。
於是我們首先建立一個ABAP資料庫表,插入兩條記錄:
再開發兩個ABAP程式ZLOCK1和ZLOCK2,分別按照Z01, Z02和Z02, Z01的順序使用SELECT SINGLE FOR UPDATE向資料庫發起讀取請求。
開啟兩個SAPGUI視窗,按照下面的步驟執行這兩個程式,即可產生死鎖。
(1) 以除錯模式單步執行程式ZLOCK1,成功執行完SELECT SINGLE FOR UPDATE .. WHERE object_id = 'Z01', 意味著此時程式ZLOCK1在資料庫層面對Z01這條記錄設定了一把鎖。
(2) 切換到另一個SAPGUI視窗,執行程式ZLOCK2, 單步除錯執行完語句SELECT SINGLE FOR UPDATE .. WHERE object_id = 'Z02',即此時程式ZLOCK2在資料庫層面對Z02成功上鎖。
(3) 再回到視窗1,繼續除錯程式ZLOCK1,此時偵錯程式會阻塞,因為ZLOCK1試圖對Z02上鎖,而此時程式ZLOCK2持有記錄Z02的鎖。程式ZLOCK1處於阻塞狀態,等待ZLOCK2釋放對記錄Z02設定的鎖。
(3) 回到視窗2,繼續在除錯模式下執行ZLOCK2程式第12行語句。此時程式ZLOCK2試圖對記錄Z01上鎖,但該記錄已經被程式ZLOCK1鎖住了,程式ZLOCK2只好等待程式ZLOCK1釋放對記錄Z01的鎖;而程式ZLOCK1永遠也不可能釋放對記錄Z01上的鎖,因為此刻它已經處於阻塞狀態了,在等待程式ZLOCK2釋放對記錄Z02的鎖。
此時ABAP Kernel檢測到程式ZLOCK2發生了死鎖,丟擲一個執行時異常,結束了ZLOCK2的執行。程式ZLOCK2持有對記錄Z02的鎖也自動釋放了。最後的結果是,程式ZLOCK1成功地對記錄Z02上了鎖。
在事務碼ST22裡,我們能觀察到這次死鎖的詳情。
ABAP Kernel這種檢測到死鎖發生後,立刻終止程式執行的方式,節省了應用開發人員通過閱讀程式碼去分析死鎖的時間。
回到Jerry之前的Java程式,執行之後兩個執行緒陷入死鎖,在控制檯上沒有列印任何可供排錯的資訊。然而JDK本身提供了方便的檢查Java應用死鎖狀態的命令列工具:jstack.
命令列執行jstack,傳入Java程式的程式id,如果該程式發生了死鎖,該工具會列印出程式裡具體是哪一行程式碼,在試圖對何種資源進行上鎖操作時出現的死鎖,從而幫助開發人員迅速定位到邏輯上存在缺陷的程式碼位置。
希望本文這個小小的例子能幫助大家回憶起死鎖這個基礎知識點,感謝閱讀。
要獲取更多Jerry的原創文章,請關注公眾號"汪子熙":
來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/24475491/viewspace-2689681/,如需轉載,請註明出處,否則將追究法律責任。
相關文章
- 程式設計面試題:編寫一個會造成資料庫死鎖的應用程式設計面試題資料庫
- 面試問題 - 只用位操作在ABAP裡實現a+b面試
- 關於 SAP HANA 資料庫的死鎖問題(deadlock)資料庫
- SAP ABAP Netweaver 裡的 ABAP 會話概念會話
- 在SAP雲平臺ABAP程式設計環境上編寫第一段ABAP程式程式設計
- ABAP 真的會過時嗎?聊聊 ABAP 的過去,現在和未來
- 面試:什麼是死鎖,如何避免或解決死鎖;MySQL中的死鎖現象,MySQL死鎖如何解決面試MySql
- 淺談 ABAP 程式執行時出現『記憶體耗盡』錯誤的問題試讀版記憶體
- SAP ABAP Gateway Client 的 ABAP 實現,重用 HTTP ConnectionGatewayclientHTTP
- 如何寫一段死鎖程式碼
- MySQL死鎖系列-線上死鎖問題排查思路MySql
- 會說話的ABAP report
- Standard ABAP Debugger 和 Classic ABAP Debugger 的實現差異
- 使用jstack檢測Java應用的死鎖(deadlock)狀態JSJava
- python面試題之“該死的for迴圈系列”(二)Python面試題
- 零基礎快速學習 ABAP之一:ABAP 伺服器的架構和一個典型的 ABAP 程式結構介紹伺服器架構
- 面試官:什麼是死鎖?怎麼排查死鎖?怎麼避免死鎖?面試
- 使用ABAP實現Mock測試工具MockitoMockito
- Restful ABAP Programming模型系列二:Action和Validation的實現REST模型
- 一些通過SAP ABAP程式碼審查得出的ABAP程式設計最佳實踐程式設計
- SAP官方釋出的ABAP程式設計規範程式設計
- 如何使用 ABAP 程式消費 SAP ABAP OData 服務
- 一文讀懂大廠面試的作業系統面試題目(超詳細整理,執行緒,程式,死鎖)作業系統面試題執行緒
- 現代 ABAP 程式語言中的正規表示式
- ABAP Platform 2020 釋出的一些新功能Platform
- SAP ABAP 程式之間的呼叫
- 面試官:請用SQL模擬一個死鎖面試SQL
- Fundamental Library for ABAP 主要的組成部分概述
- ABAP Webdynpro效能測試工具Web
- 介紹一款 ABAP 程式碼搜尋工具 RS_ABAP_SOURCE_SCAN 的使用方法
- Redis面試系列:Redis實現分散式鎖Redis面試分散式
- 一個好用的SAP ABAP工作程式跟蹤工具
- 淺談Java和SAP ABAP的靜態代理和動態代理,以及ABAP面向切面程式設計的嘗試Java程式設計
- ABAP 7.55 新特性 (一)
- SAP ABAP呼叫WEBAPI(一)WebAPI
- ABAP初學者如何系統地學習ABAP程式設計?程式設計
- 2020最新支付寶高階Java現場面試37題:頁鎖+死鎖+叢集+雪崩+負載等(文末多家大廠面試題附贈)Java負載面試題
- 【面試普通人VS高手系列】死鎖的發生原因和怎麼避免面試