溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Spring數(shù)據(jù)庫異常的示例分析

發(fā)布時間:2021-07-24 14:12:35 來源:億速云 閱讀:293 作者:小新 欄目:編程語言

這篇文章給大家分享的是有關(guān)Spring數(shù)據(jù)庫異常的示例分析的內(nèi)容。小編覺得挺實用的,因此分享給大家做個參考,一起跟隨小編過來看看吧。

數(shù)據(jù)庫為:H2

Spring數(shù)據(jù)庫異常的示例分析

如果需要處理特定 SQL 異常,比如 SQL 語句錯誤,這個時候我們應該怎么辦?

查看 SQLException 源碼,我們可以發(fā)現(xiàn)兩個重要的方法。

SQLException.getErrorCode:返回數(shù)據(jù)庫特定的錯誤碼,由數(shù)據(jù)庫廠商制定,不同廠商錯誤碼不同。如重復主鍵錯誤碼在 MySQL 中是 1062,而在 Oracle 中卻是 1。

SQLException.getSQLState:返回 XOPEN 或 SQL:2003 制定的錯誤碼規(guī)范。數(shù)據(jù)庫廠商會將不同錯誤消息映射成同一個錯誤碼

所以我們可以根據(jù) SQLException.getErrorCode 處理相應的數(shù)據(jù)庫異常。

Spring數(shù)據(jù)庫異常的示例分析

由于數(shù)據(jù)庫廠商錯誤碼不相同,這就導致如果我們更換數(shù)據(jù)庫,上面判斷邏輯就必須重寫。

下面我們使用 Spring 操作數(shù)據(jù)庫。

Spring 操作數(shù)據(jù)庫

Spring數(shù)據(jù)庫異常的示例分析

使用 Spring 之后,我們不再需要強制捕獲異常。如果 SQL 語句運行存在異常,Spring 會拋出其內(nèi)置特定的異常。如上面 SQL 語句異常將會拋出 BadSqlGrammarException。除了這個異常之外,Spring 還定義很多數(shù)據(jù)庫異常。

Spring數(shù)據(jù)庫異常的示例分析

每個 Spring 數(shù)據(jù)庫異常的基類都是 DataAccessException。由于 DataAccessException 繼承自 RuntimeException,所以在這類異常無需強制捕獲。

在 Spring 中使用 SQLExceptionTranslator 進行異常轉(zhuǎn)換,默認的轉(zhuǎn)換規(guī)則會根據(jù) SQLException.getErrorCode 返回的錯誤碼進行相應的轉(zhuǎn)換。

下面我們從源碼分析轉(zhuǎn)換過程。

實現(xiàn)細節(jié)

調(diào)試 JdbcTemplate 的源碼。

Spring數(shù)據(jù)庫異常的示例分析

可以看到這里捕獲了 SQLException,轉(zhuǎn)換之后再將其拋出。

整個轉(zhuǎn)換過程,最后交給 SQLExceptionTranslator 進行轉(zhuǎn)換。

首先我們查看 SQLExceptionTranslator 類圖。

Spring數(shù)據(jù)庫異常的示例分析

可以看到其實現(xiàn)了一個抽象類以及三個子類。

Spring數(shù)據(jù)庫異常的示例分析

抽象類中會首先會使用子類轉(zhuǎn)換,若未能轉(zhuǎn)換成功,將會啟動 fallback機制,再次轉(zhuǎn)換,作為兜底。

接著我們先看下三個子類的區(qū)別。

SQLErrorCodeSQLExceptionTranslator:

默認轉(zhuǎn)換類主要根據(jù) SQLException.getErrorCode 進行轉(zhuǎn)換。默認使用 SQLExceptionSubclassTranslator 作為 fallback 對象。

SQLExceptionSubclassTranslator:

基于 JDBC 的 SQLException 標準子類判斷,如 java.sql.SQLTransientException。使用 SQLStateSQLExceptionTranslator 作為 fallback 對象。

SQLStateSQLExceptionTranslator:

基于 SQLException.getSQLState 規(guī)則判斷。

下面分析 SQLErrorCodeSQLExceptionTranslator ,其他兩個比較類似,同學們可以自己看源碼分析。

SQLErrorCodeSQLExceptionTranslator 轉(zhuǎn)換器主要根據(jù) SQLException.getErrorCode 進行判斷。Spring 默認在 org/springframework/jdbc/support/sql-error-codes.xml 歸納不同數(shù)據(jù)庫廠商相關(guān)錯誤碼。該配置文件會在第一次發(fā)生 SQL 異常時由 SQLErrorCodesFactory 進行加載,最后生成 SQLErrorCodes。

Spring數(shù)據(jù)庫異常的示例分析

另外在 SQLErrorCodes 提供擴展方法,可以根據(jù)錯誤碼轉(zhuǎn)換成自定義的異常。

最后查看 SQLErrorCodeSQLExceptionTranslator 里的轉(zhuǎn)換方法。

Spring數(shù)據(jù)庫異常的示例分析

前三個方法是 Spring 留下擴展方法,可以根據(jù)自己需求分別擴展。若都沒有實現(xiàn),將會根據(jù)錯誤碼判斷轉(zhuǎn)換成具體的異常。

Spring數(shù)據(jù)庫異常的示例分析

自定義異常轉(zhuǎn)換

上面說到 Spring 總共給我們留下三處擴展點。

繼承 SQLErrorCodeSQLExceptionTranslator,重寫 customTranslate。繼承 SQLExceptionTranslator,重寫 translate,然后在 sql-error-codes.xml注入。使用 SQLErrorCodes#customTranslations ,然后在 sql-error-codes.xml 配置相關(guān)錯誤碼轉(zhuǎn)換的規(guī)則。

第三種方式改動最小,比較簡單。首先在 classpath 下生成 sql-error-codes.xml,復制原有配置,最后配置 customTranslations 。

Spring數(shù)據(jù)庫異常的示例分析

這里需要注意的是,需要轉(zhuǎn)化的異常類型必須為 DataAccessException 子類。下面面我們自定義一個異常。

Spring數(shù)據(jù)庫異常的示例分析

總結(jié)

Spirng 異常處理將 SQL 異常轉(zhuǎn)化成內(nèi)置異常,屏蔽不同數(shù)據(jù)庫返回碼不一致的帶來的問題。

最后總結(jié)本文的知識點,希望幫助到大家。

Spring數(shù)據(jù)庫異常的示例分析

感謝各位的閱讀!關(guān)于“Spring數(shù)據(jù)庫異常的示例分析”這篇文章就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,讓大家可以學到更多知識,如果覺得文章不錯,可以把它分享出去讓更多的人看到吧!

向AI問一下細節(jié)

免責聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI