亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

SpringBoot數(shù)據(jù)庫常見錯(cuò)誤DataIntegrityViolationException的原因及解決方案

 更新時(shí)間:2025年07月14日 10:28:06   作者:碼農(nóng)阿豪@新空間  
在SpringBoot+MyBatis/MyBatis-Plus開發(fā)過程中,數(shù)據(jù)庫操作是核心部分之一,然而,開發(fā)者經(jīng)常會(huì)遇到 org.springframework.dao.DataIntegrityViolationException異常本文將通過兩個(gè)典型案例,深入分析DataIntegrityViolationException的常見原因,并提供完整的解決方案

引言

在 Spring Boot + MyBatis/MyBatis-Plus 開發(fā)過程中,數(shù)據(jù)庫操作是核心部分之一。然而,由于 SQL 語法、數(shù)據(jù)類型不匹配、約束沖突等問題,開發(fā)者經(jīng)常會(huì)遇到 org.springframework.dao.DataIntegrityViolationException 異常。

本文將通過兩個(gè)典型案例,深入分析 DataIntegrityViolationException 的常見原因,并提供完整的解決方案。同時(shí),我們會(huì)結(jié)合代碼示例,幫助讀者理解如何避免類似問題。

1. DataIntegrityViolationException 概述

DataIntegrityViolationException 是 Spring 框架在數(shù)據(jù)庫操作失敗時(shí)拋出的異常,通常由以下原因引起:

  1. 數(shù)據(jù)類型不匹配(如字符串插入到 INT 列)
  2. 數(shù)據(jù)截?cái)啵ㄈ绯L(zhǎng)字符串插入 VARCHAR(10))
  3. 違反約束(如 NOT NULL 列插入了 NULL)
  4. SQL 語法錯(cuò)誤(如 UPDATE 語句缺少 WHERE 條件)

接下來,我們通過兩個(gè)實(shí)際案例進(jìn)行分析。

2. 案例 1:SQL 語法錯(cuò)誤導(dǎo)致的數(shù)據(jù)類型轉(zhuǎn)換異常

問題描述

在執(zhí)行以下 MyBatis XML 映射的 SQL 時(shí),報(bào)錯(cuò):

org.springframework.dao.DataIntegrityViolationException: 
### Error updating database. Cause: com.mysql.cj.jdbc.exceptions.MysqlDataTruncation: 
Data truncation: Truncated incorrect DOUBLE value: 'eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9...'

錯(cuò)誤 SQL:

<update id="updateCookie">
    update task_management set cookie = #{cookie}
    <if test="creatorId != null">
        and creator_id = #{creatorId}
    </if>
</update>

問題分析

錯(cuò)誤原因:

  • SQL 語句缺少 WHERE 關(guān)鍵字,導(dǎo)致 and creator_id = #{creatorId} 被解析為 cookie = (#{cookie} and creator_id = #{creatorId})。
  • MySQL 嘗試將 JWT Token(字符串)和 creatorId 進(jìn)行邏輯運(yùn)算,導(dǎo)致類型轉(zhuǎn)換失敗。

錯(cuò)誤 SQL 實(shí)際執(zhí)行情況:

-- 錯(cuò)誤的 SQL
UPDATE task_management SET cookie = ('eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9...' AND creator_id = 123);

-- 正確的 SQL 應(yīng)該是
UPDATE task_management SET cookie = 'eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9...' WHERE creator_id = 123;

解決方案

方法 1:使用 <where> 標(biāo)簽

<update id="updateCookie">
    UPDATE task_management
    SET cookie = #{cookie}
    <where>
        <if test="creatorId != null">
            AND creator_id = #{creatorId}
        </if>
    </where>
</update>

方法 2:使用 <trim> 標(biāo)簽

<update id="updateCookie">
    UPDATE task_management
    SET cookie = #{cookie}
    <trim prefix="WHERE" prefixOverrides="AND">
        <if test="creatorId != null">
            AND creator_id = #{creatorId}
        </if>
    </trim>
</update>

方法 3:使用注解方式

@Update("<script>" +
        "UPDATE task_management SET cookie = #{cookie}" +
        "<where>" +
        "   <if test='creatorId != null'>AND creator_id = #{creatorId}</if>" +
        "</where>" +
        "</script>")
void updateCookie(@Param("cookie") String cookie, @Param("creatorId") Long creatorId);

最佳實(shí)踐

  • 始終在 UPDATE/DELETE 語句中使用 WHERE 條件,避免全表更新。
  • 使用 MyBatis 動(dòng)態(tài) SQL 標(biāo)簽(<where>、<trim>)防止語法錯(cuò)誤。
  • 啟用 SQL 日志,檢查實(shí)際執(zhí)行的 SQL:
logging.level.org.mybatis=DEBUG

3. 案例 2:數(shù)據(jù)截?cái)噱e(cuò)誤(Data Truncation)

問題描述

在插入數(shù)據(jù)時(shí),報(bào)錯(cuò):

org.springframework.dao.DataIntegrityViolationException:
### Error updating database. Cause: java.sql.SQLException: 
Data truncated for column 'match_status' at row 1

錯(cuò)誤 SQL:

INSERT INTO customer_order (match_status) VALUES ('pending_verification');

問題分析

可能原因:

  • match_statusENUM,但插入了不在枚舉列表的值。
  • match_statusTINYINT,但插入了字符串或超出范圍的值(如 256)。
  • match_statusVARCHAR(10),但插入了超長(zhǎng)字符串(如 "pending_verification")。

排查方法:

  • 查看表結(jié)構(gòu):
DESC customer_order;
  • 檢查 match_status 列的定義:
SHOW CREATE TABLE customer_order;

解決方案

情況 1:match_status 是 ENUM

如果 match_statusENUM('pending', 'success', 'failed'),則只能插入這三個(gè)值之一:

ALTER TABLE customer_order 
MODIFY COLUMN match_status ENUM('pending', 'success', 'failed') NOT NULL;

代碼調(diào)整:

customerOrder.setMatchStatus("success"); // 正確
// customerOrder.setMatchStatus("invalid_status"); // 錯(cuò)誤

情況 2:match_status 是 TINYINT

如果 match_statusTINYINT(0-255),則不能插入字符串或超出范圍的值:

ALTER TABLE customer_order 
MODIFY COLUMN match_status TINYINT UNSIGNED NOT NULL DEFAULT 0;

代碼調(diào)整:

customerOrder.setMatchStatus(1); // 正確
// customerOrder.setMatchStatus(256); // 錯(cuò)誤(超出范圍)
// customerOrder.setMatchStatus("pending"); // 錯(cuò)誤(類型不匹配)

情況 3:match_status 是 VARCHAR 但長(zhǎng)度不足

如果 match_statusVARCHAR(10),但插入了更長(zhǎng)的字符串:

ALTER TABLE customer_order 
MODIFY COLUMN match_status VARCHAR(50); -- 擴(kuò)大長(zhǎng)度

代碼調(diào)整:

customerOrder.setMatchStatus("pending"); // 正確
// customerOrder.setMatchStatus("pending_verification"); // 可能被截?cái)?

最佳實(shí)踐

  • 在數(shù)據(jù)庫設(shè)計(jì)階段明確字段類型和約束。
  • 在代碼中校驗(yàn)數(shù)據(jù)是否符合數(shù)據(jù)庫約束。
  • 使用日志監(jiān)控 SQL 執(zhí)行情況:
logging.level.org.springframework.jdbc=DEBUG

4. 總結(jié)與預(yù)防措施

問題類型錯(cuò)誤示例解決方案
SQL 語法錯(cuò)誤UPDATE ... SET col = val AND ...使用 <where> 標(biāo)簽
ENUM 不匹配插入不在枚舉列表的值修改數(shù)據(jù)庫或代碼
數(shù)值超出范圍TINYINT 插入 256檢查數(shù)據(jù)庫范圍
字符串超長(zhǎng)VARCHAR(10) 插入 20 字符擴(kuò)大列長(zhǎng)度或截?cái)鄶?shù)據(jù)

通用調(diào)試技巧

查看數(shù)據(jù)庫表結(jié)構(gòu):

DESC table_name;
SHOW CREATE TABLE table_name;

啟用 SQL 日志:

logging.level.org.mybatis=DEBUG
logging.level.org.springframework.jdbc=DEBUG

在代碼中校驗(yàn)數(shù)據(jù):

if (value.length() > 10) {
    throw new IllegalArgumentException("字段超長(zhǎng)");
}

5. 結(jié)語

DataIntegrityViolationException 是 Spring Boot 開發(fā)中常見的數(shù)據(jù)庫異常,通常由 SQL 語法錯(cuò)誤、數(shù)據(jù)類型不匹配或約束沖突引起。通過本文的分析和解決方案,希望讀者能夠更高效地定位和修復(fù)類似問題。

關(guān)鍵點(diǎn)回顧:

  1. SQL 語法檢查:確保 UPDATE/DELETE 語句有正確的 WHERE 條件。
  2. 數(shù)據(jù)類型匹配:確保 Java 代碼傳入的值與數(shù)據(jù)庫列類型一致。
  3. 約束校驗(yàn):避免插入 NULL、超長(zhǎng)或不符合 ENUM 的值。

以上就是SpringBoot數(shù)據(jù)庫常見錯(cuò)誤DataIntegrityViolationException的原因及解決方案的詳細(xì)內(nèi)容,更多關(guān)于SpringBoot數(shù)據(jù)庫錯(cuò)誤DataIntegrityViolationException的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • Spring重試支持Spring Retry的方法

    Spring重試支持Spring Retry的方法

    本篇文章主要介紹了Spring重試支持Spring Retry的方法,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2018-04-04
  • 解讀String字符串導(dǎo)致的JVM內(nèi)存泄漏問題

    解讀String字符串導(dǎo)致的JVM內(nèi)存泄漏問題

    這篇文章主要介紹了解讀String字符串導(dǎo)致的JVM內(nèi)存泄漏問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-07-07
  • MyBatisPlus之id生成策略的方法

    MyBatisPlus之id生成策略的方法

    本文主要介紹了MyBatisPlus之id生成策略的方法,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-03-03
  • Java設(shè)計(jì)模式中的建造者模式詳解

    Java設(shè)計(jì)模式中的建造者模式詳解

    這篇文章主要介紹了Java設(shè)計(jì)模式中的建造者模式詳解,建造者模式使我們?nèi)粘9ぷ髦斜容^常見的一種設(shè)計(jì)模式,和工廠模式一樣屬于創(chuàng)建型設(shè)計(jì)模式,用于解耦對(duì)象創(chuàng)建和對(duì)象使用的邏輯,需要的朋友可以參考下
    2023-12-12
  • SpringBoot任務(wù)之詳解郵件任務(wù)

    SpringBoot任務(wù)之詳解郵件任務(wù)

    今天給大家整理的文章是SpringBoot郵件任務(wù)的相關(guān)知識(shí)點(diǎn),文中有非常詳細(xì)的介紹及代碼示例,對(duì)正在學(xué)習(xí)SpringBoot任務(wù)的小伙伴們很有幫助,需要的朋友可以參考下
    2021-06-06
  • 一文帶你深入了解Java8 Stream流式編程

    一文帶你深入了解Java8 Stream流式編程

    在實(shí)際項(xiàng)目當(dāng)中,若能熟練使用Java8 的Stream流特性進(jìn)行開發(fā),就比較容易寫出簡(jiǎn)潔優(yōu)雅的代碼。本文主要就是基于實(shí)際項(xiàng)目常用的Stream Api流式處理總結(jié),希望對(duì)大家有所幫助
    2023-04-04
  • 詳解java設(shè)計(jì)模式中的門面模式

    詳解java設(shè)計(jì)模式中的門面模式

    門面模式又叫外觀模式(Facade?Pattern),主要用于隱藏系統(tǒng)的復(fù)雜性,并向客戶端提供了一個(gè)客戶端可以訪問系統(tǒng)的接口,本文通過實(shí)例代碼給大家介紹下java門面模式的相關(guān)知識(shí),感興趣的朋友一起看看吧
    2022-02-02
  • Java實(shí)現(xiàn)瀏覽器大文件上傳的示例詳解

    Java實(shí)現(xiàn)瀏覽器大文件上傳的示例詳解

    文件上傳是許多項(xiàng)目都有的功能,用戶上傳小文件速度一般都很快,但如果是大文件幾個(gè)g,幾十個(gè)g的時(shí)候,上傳了半天,馬上就要完成的時(shí)候,網(wǎng)絡(luò)波動(dòng)一下,文件又要重新上傳,所以本文給大家介紹了Java實(shí)現(xiàn)瀏覽器大文件上傳的示例,需要的朋友可以參考下
    2024-07-07
  • java獲取本月日歷表的方法

    java獲取本月日歷表的方法

    這篇文章主要為大家詳細(xì)介紹了java獲取本月日歷表的方法,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-06-06
  • JavaBean實(shí)體類處理外鍵過程解析

    JavaBean實(shí)體類處理外鍵過程解析

    這篇文章主要介紹了JavaBean實(shí)體類處理外鍵過程解析,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-07-07

最新評(píng)論