欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

深度解析MyBatis?動(dòng)態(tài)?SQL?與緩存機(jī)制

 更新時(shí)間:2025年06月28日 08:55:59   作者:晴空月明  
本文從動(dòng)態(tài)SQL核心語(yǔ)法、緩存實(shí)現(xiàn)原理、性能優(yōu)化及面試高頻問(wèn)題四個(gè)維度,結(jié)合源碼與工程實(shí)踐,系統(tǒng)解析MyBatis的核心特性與最佳實(shí)踐,感興趣的朋友一起看看吧

在Java持久層技術(shù)體系中,MyBatis憑借其靈活的SQL映射和強(qiáng)大的動(dòng)態(tài)SQL能力,成為企業(yè)級(jí)應(yīng)用開發(fā)的首選框架。本文從動(dòng)態(tài)SQL核心語(yǔ)法、緩存實(shí)現(xiàn)原理、性能優(yōu)化及面試高頻問(wèn)題四個(gè)維度,結(jié)合源碼與工程實(shí)踐,系統(tǒng)解析MyBatis的核心特性與最佳實(shí)踐。

一、動(dòng)態(tài)SQL核心語(yǔ)法與實(shí)現(xiàn)原理

1.1 動(dòng)態(tài)SQL標(biāo)簽體系

標(biāo)簽作用示例場(chǎng)景
<if>條件判斷,按需拼接SQL片段動(dòng)態(tài)查詢(如多條件篩選)
<choose>類似于Java的switch語(yǔ)句,多選一執(zhí)行單條件查詢(不同條件互斥)
<where>智能處理WHERE子句,自動(dòng)剔除多余的AND/OR動(dòng)態(tài)WHERE條件
<set>智能處理UPDATE語(yǔ)句,自動(dòng)剔除多余的逗號(hào)動(dòng)態(tài)更新(部分字段更新)
<foreach>遍歷集合,生成批量SQL批量插入、IN條件查詢
<trim>自定義前綴、后綴處理,可替代<where>、<set>高級(jí)SQL片段處理

1.2 動(dòng)態(tài)SQL執(zhí)行流程

關(guān)鍵步驟解析:

  • SQL節(jié)點(diǎn)解析
    • XML配置中的動(dòng)態(tài)標(biāo)簽(如<if>)被解析為SqlNode對(duì)象(如IfSqlNode)。
  • OGNL表達(dá)式計(jì)算
    • 使用OGNL(Object Graph Navigation Language)計(jì)算動(dòng)態(tài)條件(如#{username} != null)。
  • 參數(shù)綁定
    • 通過(guò)TypeHandler將Java對(duì)象轉(zhuǎn)換為JDBC類型(如String → VARCHAR)。

1.3 高級(jí)應(yīng)用:自定義SQL提供器

1. 使用@Provider注解

@SelectProvider(type = UserSqlProvider.class, method = "selectByCondition")  
List<User> selectByCondition(Map<String, Object> params);  
// 自定義SQL提供器  
public class UserSqlProvider {  
    public String selectByCondition(Map<String, Object> params) {  
        SQL sql = new SQL();  
        sql.SELECT("*").FROM("users");  
        if (params.containsKey("username")) {  
            sql.WHERE("username = #{username}");  
        }  
        if (params.containsKey("age")) {  
            sql.WHERE("age >= #{age}");  
        }  
        return sql.toString();  
    }  
}  

2. 流式SQL構(gòu)建(SQL類)

String sql = new SQL()  
    .SELECT("id", "username")  
    .FROM("users")  
    .WHERE("status = 'ACTIVE'")  
    .ORDER_BY("create_time DESC")  
    .toString();  

二、緩存機(jī)制深度解析

2.1 一級(jí)緩存(本地緩存)

1. 核心特性

  • 作用域SqlSession級(jí)別(同一個(gè)會(huì)話內(nèi)共享)。
  • 生命周期:與SqlSession一致,會(huì)話關(guān)閉時(shí)緩存清空。
  • 實(shí)現(xiàn)類PerpetualCache(基于HashMap)。

2. 源碼關(guān)鍵邏輯

public class DefaultSqlSession implements SqlSession {  
    private final Executor executor;  
    @Override  
    public <T> T selectOne(String statement, Object parameter) {  
        List<T> list = this.selectList(statement, parameter);  
        // 一級(jí)緩存邏輯在Executor中實(shí)現(xiàn)  
        return list.isEmpty() ? null : list.get(0);  
    }  
}  
public class BaseExecutor implements Executor {  
    private final PerpetualCache localCache;  
    @Override  
    public <E> List<E> query(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler) throws SQLException {  
        BoundSql boundSql = ms.getBoundSql(parameter);  
        CacheKey key = createCacheKey(ms, parameter, rowBounds, boundSql);  
        return query(ms, parameter, rowBounds, resultHandler, key, boundSql);  
    }  
    @Override  
    public <E> List<E> query(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, CacheKey key, BoundSql boundSql) throws SQLException {  
        // 先從一級(jí)緩存獲取  
        List<E> list = (List<E>) localCache.getObject(key);  
        if (list != null) {  
            handleLocallyCachedOutputParameters(ms, key, parameter, boundSql);  
            return list;  
        } else {  
            // 緩存未命中,執(zhí)行數(shù)據(jù)庫(kù)查詢  
            list = queryFromDatabase(ms, parameter, rowBounds, resultHandler, key, boundSql);  
            return list;  
        }  
    }  
}  

2.2 二級(jí)緩存(全局緩存)

1. 核心特性

  • 作用域namespace級(jí)別(跨會(huì)話共享)。
  • 配置方式
    <cache eviction="LRU" flushInterval="60000" size="512" readOnly="true"/>  
  • 默認(rèn)實(shí)現(xiàn)PerpetualCache(可替換為Redis、Ehcache等)。

2. 緩存配置參數(shù)

參數(shù)作用
eviction緩存淘汰策略(LRU/FIFO/SOFT/WEAK)
flushInterval刷新間隔(毫秒,默認(rèn)不刷新)
size緩存最大容量(元素個(gè)數(shù))
readOnly是否只讀(true則返回緩存對(duì)象的引用,性能更高)

2.3 緩存工作流程

關(guān)鍵注意點(diǎn):

  • 緩存失效
    增刪改操作(INSERT/UPDATE/DELETE)默認(rèn)會(huì)清空所在namespace的二級(jí)緩存。
  • 嵌套查詢
    嵌套查詢(<association>、<collection>)可能導(dǎo)致二級(jí)緩存失效(取決于useCache屬性)。

三、緩存集成與性能優(yōu)化

3.1 第三方緩存集成(Redis示例)

1. 添加依賴

<dependency>  
    <groupId>org.mybatis.caches</groupId>  
    <artifactId>mybatis-redis</artifactId>  
    <version>1.0.0-beta2</version>  
</dependency>  

2. 配置Redis緩存

<cache type="org.mybatis.caches.redis.RedisCache"/>  
<!-- redis.properties -->  
host=127.0.0.1  
port=6379  
timeout=2000  

3.2 性能優(yōu)化策略

1. 合理使用緩存級(jí)別

  • 一級(jí)緩存:默認(rèn)開啟,適合短期高頻查詢(如同一請(qǐng)求內(nèi)多次查詢相同數(shù)據(jù))。
  • 二級(jí)緩存:需顯式配置,適合全局共享且讀多寫少的數(shù)據(jù)(如字典表、配置信息)。

2. 緩存參數(shù)調(diào)優(yōu)

<!-- 高并發(fā)場(chǎng)景優(yōu)化配置 -->  
<cache  
    eviction="LRU"  
    flushInterval="300000" <!-- 5分鐘刷新一次 -->  
    size="2048"          <!-- 增大緩存容量 -->  
    readOnly="true"/>    <!-- 只讀模式提升性能 -->  

3. 避免緩存穿透與雪崩

  • 緩存穿透
    查詢不存在的數(shù)據(jù)導(dǎo)致每次都訪問(wèn)數(shù)據(jù)庫(kù),可通過(guò)布隆過(guò)濾器或緩存空值解決。
  • 緩存雪崩
    大量緩存同時(shí)失效導(dǎo)致瞬間數(shù)據(jù)庫(kù)壓力劇增,可通過(guò)設(shè)置隨機(jī)過(guò)期時(shí)間避免。

四、面試高頻問(wèn)題深度解析

4.1 基礎(chǔ)概念類問(wèn)題

Q:MyBatis動(dòng)態(tài)SQL與Hibernate Criteria API的區(qū)別?A:

維度MyBatis動(dòng)態(tài)SQLHibernate Criteria API
SQL控制完全手動(dòng)控制,靈活性高通過(guò)API生成,靈活性低
學(xué)習(xí)成本較低(熟悉XML標(biāo)簽即可)較高(需掌握對(duì)象化查詢API)
性能接近原生SQL,性能優(yōu)化空間大可能生成冗余SQL,優(yōu)化難度高
適用場(chǎng)景復(fù)雜SQL場(chǎng)景(如多表關(guān)聯(lián)、嵌套查詢)簡(jiǎn)單增刪改查場(chǎng)景

Q:MyBatis一級(jí)緩存與二級(jí)緩存的區(qū)別?A:

特性一級(jí)緩存二級(jí)緩存
作用域SqlSession級(jí)別Namespace級(jí)別
生命周期會(huì)話關(guān)閉后失效應(yīng)用啟動(dòng)到關(guān)閉
默認(rèn)開啟
緩存共享同一個(gè)會(huì)話內(nèi)共享跨會(huì)話共享
實(shí)現(xiàn)類PerpetualCache可自定義(如RedisCache)

4.2 實(shí)現(xiàn)原理類問(wèn)題

Q:MyBatis如何實(shí)現(xiàn)動(dòng)態(tài)SQL的條件判斷?A:

  • 通過(guò)OGNL表達(dá)式計(jì)算條件(如#{username} != null)。
  • 解析為對(duì)應(yīng)的SqlNode實(shí)現(xiàn)類(如IfSqlNode)。
  • 在SQL執(zhí)行時(shí)動(dòng)態(tài)決定是否包含該SQL片段。

Q:二級(jí)緩存的嵌套查詢會(huì)導(dǎo)致什么問(wèn)題?如何解決?A:

  • 問(wèn)題:嵌套查詢默認(rèn)不使用二級(jí)緩存,可能導(dǎo)致重復(fù)查詢數(shù)據(jù)庫(kù)。
  • 解決方案

    設(shè)置useCache="true"flushCache="false"。

    使用<resultMap>的嵌套映射替代嵌套查詢。

4.3 實(shí)戰(zhàn)調(diào)優(yōu)類問(wèn)題

Q:如何解決MyBatis緩存與數(shù)據(jù)庫(kù)一致性問(wèn)題?A:

  • 更新策略
    • 增刪改操作后強(qiáng)制刷新緩存(默認(rèn)行為)。
    • 設(shè)置合理的緩存過(guò)期時(shí)間(如5分鐘)。
  • 讀寫分離場(chǎng)景
    • 主庫(kù)寫操作后立即刷新緩存。
    • 從庫(kù)讀操作使用緩存,通過(guò)數(shù)據(jù)庫(kù)主從同步保證最終一致性。

Q:MyBatis動(dòng)態(tài)SQL中<where>標(biāo)簽與<trim>標(biāo)簽的區(qū)別?A:

  • <where>
    自動(dòng)添加WHERE關(guān)鍵字,并剔除多余的AND/OR。
  • <trim>
    可自定義前綴、后綴處理,如:
    <trim prefix="WHERE" prefixOverrides="AND |OR ">  
        ...  
    </trim>  
    更靈活,可替代<where>標(biāo)簽。

總結(jié):動(dòng)態(tài)SQL與緩存的最佳實(shí)踐

動(dòng)態(tài)SQL設(shè)計(jì)原則

  • 簡(jiǎn)潔優(yōu)先:避免過(guò)度復(fù)雜的動(dòng)態(tài)SQL,優(yōu)先使用<if>、<where>等基礎(chǔ)標(biāo)簽。
  • 參數(shù)校驗(yàn):在Java代碼中進(jìn)行參數(shù)校驗(yàn),避免在動(dòng)態(tài)SQL中處理復(fù)雜邏輯。
  • SQL片段復(fù)用:使用<sql>標(biāo)簽定義公共SQL片段,通過(guò)<include>復(fù)用。

緩存使用策略

  • 讀多寫少場(chǎng)景:?jiǎn)⒂枚?jí)緩存,如字典表、配置信息。
  • 寫操作頻繁場(chǎng)景:禁用二級(jí)緩存,避免頻繁刷新影響性能。
  • 分布式環(huán)境:使用Redis等分布式緩存替代默認(rèn)實(shí)現(xiàn),保證跨節(jié)點(diǎn)緩存一致性。

通過(guò)系統(tǒng)化掌握MyBatis動(dòng)態(tài)SQL與緩存機(jī)制的核心原理及最佳實(shí)踐,面試者可在回答中精準(zhǔn)匹配問(wèn)題需求,例如分析“如何優(yōu)化復(fù)雜查詢性能”時(shí),能結(jié)合動(dòng)態(tài)SQL優(yōu)化與緩存策略,展現(xiàn)對(duì)持久層技術(shù)的深度理解與工程實(shí)踐能力。

到此這篇關(guān)于MyBatis 動(dòng)態(tài) SQL 與緩存機(jī)制深度解析的文章就介紹到這了,更多相關(guān)MyBatis 動(dòng)態(tài) SQL內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • SpringBoot基于AbstractRoutingDataSource實(shí)現(xiàn)多數(shù)據(jù)源動(dòng)態(tài)切換

    SpringBoot基于AbstractRoutingDataSource實(shí)現(xiàn)多數(shù)據(jù)源動(dòng)態(tài)切換

    本文主要介紹了SpringBoot基于AbstractRoutingDataSource實(shí)現(xiàn)多數(shù)據(jù)源動(dòng)態(tài)切換,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2022-05-05
  • 基于SPRINGBOOT配置文件占位符過(guò)程解析

    基于SPRINGBOOT配置文件占位符過(guò)程解析

    這篇文章主要介紹了基于SPRINGBOOT配置文件占位符過(guò)程解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2019-12-12
  • Spring Boot讀取配置屬性常用方法解析

    Spring Boot讀取配置屬性常用方法解析

    這篇文章主要介紹了Spring Boot讀取配置屬性常用方法解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-09-09
  • SpringMVC如何用Post方式重定向

    SpringMVC如何用Post方式重定向

    這篇文章主要介紹了SpringMVC如何用Post方式重定向,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-10-10
  • 淺談java定時(shí)器的發(fā)展歷程

    淺談java定時(shí)器的發(fā)展歷程

    這篇文章主要介紹了淺談java定時(shí)器的發(fā)展歷程,具有一定借鑒價(jià)值,需要的朋友可以參考下。
    2017-12-12
  • java數(shù)據(jù)庫(kù)開發(fā)之JDBC的完整封裝兼容多種數(shù)據(jù)庫(kù)

    java數(shù)據(jù)庫(kù)開發(fā)之JDBC的完整封裝兼容多種數(shù)據(jù)庫(kù)

    這篇文章主要介紹了java數(shù)據(jù)庫(kù)開發(fā)之JDBC的完整封裝兼容多種數(shù)據(jù)庫(kù),需要的朋友可以參考下
    2020-02-02
  • SpringBoot?Test的webEnvironment源碼解讀

    SpringBoot?Test的webEnvironment源碼解讀

    這篇文章主要為大家介紹了SpringBoot?Test的webEnvironment源碼解讀,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-09-09
  • Spring復(fù)雜對(duì)象創(chuàng)建的方式小結(jié)

    Spring復(fù)雜對(duì)象創(chuàng)建的方式小結(jié)

    這篇文章主要介紹了Spring復(fù)雜對(duì)象創(chuàng)建的三種方式,現(xiàn)在使用Spring如何創(chuàng)建這種類型的對(duì)象?Spring中提供了三種方法來(lái)創(chuàng)建復(fù)雜對(duì)象,需要的朋友可以參考下
    2022-01-01
  • 關(guān)于Java中的klass和class

    關(guān)于Java中的klass和class

    這篇文章主要介紹了關(guān)于Java中klass和class的區(qū)別,vm加載的字節(jié)碼,也就是.class文件,被加載到方法區(qū)里面,叫Kclass,是一個(gè)C++對(duì)象,含有類的信息、虛方法表等,需要的朋友可以參考下
    2023-08-08
  • Java實(shí)現(xiàn)動(dòng)態(tài)代理

    Java實(shí)現(xiàn)動(dòng)態(tài)代理

    本文給大家介紹的是java使用動(dòng)態(tài)代理類實(shí)現(xiàn)動(dòng)態(tài)代理的方法和示例,這里推薦給大家,有需要的小伙伴參考下吧
    2015-02-02

最新評(píng)論