ABAP面試題系列:寫一組會出現(xiàn)死鎖(Deadlock)的ABAP程序

      網(wǎng)友投稿 826 2022-05-30

      我們在計算機操作系統(tǒng)這門專業(yè)課上,學過死鎖(Deadlock)的概念:兩個或兩個以上的進程(或線程)在執(zhí)行過程中,由于競爭資源而造成的一種阻塞的現(xiàn)象,稱為死鎖。若無外力干預,這些處于死鎖狀態(tài)的進程將永遠處于互相等待的阻塞狀態(tài)中。

      正好我兒子走到我電腦前看到文章標題,好奇地問我什么是死鎖。我解釋道,“假設你和白妹妹(他的玩伴)手上都有一張奧特曼白金卡,你特別想要白妹妹手上那張白金卡,白妹妹也特別想要你手上那張白金卡。你們都想讓對方把他/她手上那張卡送給你們,但你們都舍不得把自己手上那張卡送出去。這就是死鎖。”

      兒子又問,那這種情況咋辦。

      我說,只有靠大人的介入。比如你老爸出馬,把你們手中兩張卡都沒收了,等我玩夠了再還給你們,這樣你們就不會死鎖了。

      言歸正傳,在使用ABAP答這道面試題之前,我們先看看如何用Java編寫一個會出現(xiàn)死鎖的程序。

      不到40行代碼就完成任務。為了便于ABAP從業(yè)人員理解,沒有使用Java里的Lambda表達式,否則代碼可以更短。

      該程序邏輯如下:

      線程1持有資源1,試圖持有資源2;

      線程2持有資源2,試圖持有資源1;

      死鎖發(fā)生;

      現(xiàn)在進入ABAP部分。

      ABAP幫助文檔提到,使用SELECT SINGLE FOR UPDATE讀取單條記錄時,會自動在數(shù)據(jù)庫層面為該條記錄設置一把鎖(Exclusive lock,有的中文文檔翻譯成排他鎖)。如果上鎖操作會導致死鎖發(fā)生,則會拋出異常。

      于是我們首先創(chuàng)建一個ABAP數(shù)據(jù)庫表,插入兩條記錄:

      再開發(fā)兩個ABAP程序ZLOCK1和ZLOCK2,分別按照Z01, Z02和Z02, Z01的順序使用SELECT SINGLE FOR UPDATE向數(shù)據(jù)庫發(fā)起讀取請求。

      開啟兩個SAPGUI窗口,按照下面的步驟執(zhí)行這兩個程序,即可產(chǎn)生死鎖。

      (1) 以調(diào)試模式單步執(zhí)行程序ZLOCK1,成功執(zhí)行完SELECT SINGLE FOR UPDATE … WHERE object_id = ‘Z01’, 意味著此時程序ZLOCK1在數(shù)據(jù)庫層面對Z01這條記錄設置了一把鎖。

      (2) 切換到另一個SAPGUI窗口,執(zhí)行程序ZLOCK2, 單步調(diào)試執(zhí)行完語句SELECT SINGLE FOR UPDATE … WHERE object_id = ‘Z02’,即此時程序ZLOCK2在數(shù)據(jù)庫層面對Z02成功上鎖。

      (3) 再回到窗口1,繼續(xù)調(diào)試程序ZLOCK1,此時調(diào)試器會阻塞,因為ZLOCK1試圖對Z02上鎖,而此時程序ZLOCK2持有記錄Z02的鎖。程序ZLOCK1處于阻塞狀態(tài),等待ZLOCK2釋放對記錄Z02設置的鎖。

      (3) 回到窗口2,繼續(xù)在調(diào)試模式下執(zhí)行ZLOCK2程序第12行語句。此時程序ZLOCK2試圖對記錄Z01上鎖,但該記錄已經(jīng)被程序ZLOCK1鎖住了,程序ZLOCK2只好等待程序ZLOCK1釋放對記錄Z01的鎖;而程序ZLOCK1永遠也不可能釋放對記錄Z01上的鎖,因為此刻它已經(jīng)處于阻塞狀態(tài)了,在等待程序ZLOCK2釋放對記錄Z02的鎖。

      此時ABAP Kernel檢測到程序ZLOCK2發(fā)生了死鎖,拋出一個運行時異常,結(jié)束了ZLOCK2的執(zhí)行。程序ZLOCK2持有對記錄Z02的鎖也自動釋放了。最后的結(jié)果是,程序ZLOCK1成功地對記錄Z02上了鎖。

      在事務碼ST22里,我們能觀察到這次死鎖的詳情。

      ABAP Kernel這種檢測到死鎖發(fā)生后,立刻終止程序運行的方式,節(jié)省了應用開發(fā)人員通過閱讀代碼去分析死鎖的時間。

      回到Jerry之前的Java程序,運行之后兩個線程陷入死鎖,在控制臺上沒有打印任何可供排錯的信息。然而JDK本身提供了方便的檢查Java應用死鎖狀態(tài)的命令行工具:jstack.

      命令行執(zhí)行jstack,傳入Java程序的進程id,如果該程序發(fā)生了死鎖,該工具會打印出程序里具體是哪一行代碼,在試圖對何種資源進行上鎖操作時出現(xiàn)的死鎖,從而幫助開發(fā)人員迅速定位到邏輯上存在缺陷的代碼位置。

      希望本文這個小小的例子能幫助大家回憶起死鎖這個基礎知識點,感謝閱讀。

      ABAP面試題系列:寫一組會出現(xiàn)死鎖(Deadlock)的ABAP程序

      ABAP HTTP Java SQL 容器

      版權(quán)聲明:本文內(nèi)容由網(wǎng)絡用戶投稿,版權(quán)歸原作者所有,本站不擁有其著作權(quán),亦不承擔相應法律責任。如果您發(fā)現(xiàn)本站中有涉嫌抄襲或描述失實的內(nèi)容,請聯(lián)系我們jiasou666@gmail.com 處理,核實后本網(wǎng)站將在24小時內(nèi)刪除侵權(quán)內(nèi)容。

      上一篇:【每天進步一點點】【Atlas 200 DK】基于YOLOv3_Resnet18的行人檢測
      下一篇:《R數(shù)據(jù)科學實戰(zhàn):工具詳解與案例分析 》 —1.1.2 read.delim/delim2—特定分隔符數(shù)據(jù)讀取
      相關(guān)文章
      国产亚洲一区二区三区在线| 亚洲性无码AV中文字幕| 亚洲精品456人成在线| 亚洲色成人网一二三区| 亚洲欧洲日韩不卡| 亚洲AV永久纯肉无码精品动漫| 久久久久国产亚洲AV麻豆| 亚洲伦乱亚洲h视频| 亚洲国产高清在线一区二区三区 | 亚洲AV无码久久精品蜜桃| 亚洲日本乱码在线观看| 国产亚洲大尺度无码无码专线| 久久精品国产亚洲Aⅴ香蕉| 亚洲综合无码精品一区二区三区| 国产亚洲精品a在线观看| 亚洲日产韩国一二三四区| 国产成A人亚洲精V品无码性色| 亚洲AV综合色区无码一区| 久久精品国产亚洲AV麻豆不卡 | 亚洲国产精品无码久久久秋霞2 | 国产精品久久久亚洲| 亚洲av无码一区二区三区乱子伦| 亚洲第一精品在线视频| 91亚洲精品视频| 亚洲av日韩av综合| 亚洲中文字幕无码中文| 亚洲AV无码成人网站在线观看 | 国产精品亚洲综合| 亚洲国产一成久久精品国产成人综合 | 久久精品国产亚洲av瑜伽| 综合一区自拍亚洲综合图区| 国产亚洲Av综合人人澡精品| 国产av无码专区亚洲国产精品| 超清首页国产亚洲丝袜| 久久国产亚洲电影天堂| 亚洲欧洲精品久久| 中文字幕在线日亚洲9| 日韩色日韩视频亚洲网站 | 亚洲视屏在线观看| 亚洲中文字幕无码mv| 日韩成人精品日本亚洲|