Java經(jīng)典面試題匯總:Mybatis
1. MyBatis 中 #{}和 ${}的區(qū)別是什么?
#{}是預(yù)編譯處理,${}是字符替換。 在使用 #{}時,MyBatis 會將 SQL 中的 #{}替換成“?”, 配合 PreparedStatement 的 set 方法賦值,這樣可以有效的防止 SQL 注入,保證程序的運(yùn)行安全。
2. MyBatis 有幾種分頁方式?
邏輯分頁: 使用 MyBatis 自帶的 RowBounds 進(jìn)行分頁,它是一次性查詢很多數(shù)據(jù),然后在數(shù)據(jù)中再進(jìn)行檢索。
物理分頁: 自己手寫 SQL 分頁或使用分頁插件 PageHelper,去數(shù)據(jù)庫查詢指定條數(shù)的分頁數(shù)據(jù)的形式。
3. MyBatis 邏輯分頁和物理分頁的區(qū)別是什么?
邏輯分頁是一次性查詢很多數(shù)據(jù),然后再在結(jié)果中檢索分頁的數(shù)據(jù)。這樣做弊端是需要消耗大量的內(nèi)存、有內(nèi)存溢出的風(fēng)險、對數(shù)據(jù)庫壓力較大。
物理分頁是從數(shù)據(jù)庫查詢指定條數(shù)的數(shù)據(jù),彌補(bǔ)了一次性全部查出的所有數(shù)據(jù)的種種缺點(diǎn),比如需要大量的內(nèi)存,對數(shù)據(jù)庫查詢壓力較大等問題。
4. MyBatis 是否支持延遲加載?延遲加載的原理是什么?
MyBatis 支持延遲加載,設(shè)置 lazyLoadingEnabled=true 即可。 延遲加載的原理的是調(diào)用的時候觸發(fā)加載,而不是在初始化的時候就加載信息。 比如調(diào)用 a. getB(). getName(),這個時候發(fā)現(xiàn) a. getB() 的值為 null,此時會單獨(dú)觸發(fā)事先保存好的關(guān)聯(lián) B 對象的 SQL, 先查詢出來 B,然后再調(diào)用 a. setB(b),而這時候再調(diào)用 a. getB(). getName() 就有值了,這就是延遲加載的基本原理
5. 說一下 MyBatis 的一級緩存和二級緩存?
一級緩存:基于 PerpetualCache 的 HashMap 本地緩存,它的聲明周期是和 SQLSession 一致的,有多個 SQLSession 或者分布式的環(huán)境中數(shù)據(jù)庫操作,可能會出現(xiàn)臟數(shù)據(jù)。當(dāng) Session flush 或 close 之后,該 Session 中的所有 Cache 就將清空,默認(rèn)一級緩存是開啟的。
二級緩存:也是基于 PerpetualCache 的 HashMap 本地緩存,不同在于其存儲作用域?yàn)?Mapper 級別的,如果多個SQLSession之間需要共享緩存,則需要使用到二級緩存,并且二級緩存可自定義存儲源,如 Ehcache。默認(rèn)不打開二級緩存,要開啟二級緩存,使用二級緩存屬性類需要實(shí)現(xiàn) Serializable 序列化接口(可用來保存對象的狀態(tài))。 開啟二級緩存數(shù)據(jù)查詢流程:二級緩存 -> 一級緩存 -> 數(shù)據(jù)庫。 緩存更新機(jī)制:當(dāng)某一個作用域(一級緩存 Session/二級緩存 Mapper)進(jìn)行了C/U/D 操作后,默認(rèn)該作用域下所有 select 中的緩存將被 clear。
6. MyBatis 有哪些執(zhí)行器(Executor)?
MyBatis 有三種基本的Executor執(zhí)行器:
- SimpleExecutor:每執(zhí)行一次 update 或 select 就開啟一個 Statement 對象,用完立刻關(guān)閉 Statement 對象;
- ReuseExecutor:執(zhí)行 update 或 select,以 SQL 作為 key 查找 Statement 對象,存在就使用,不存在就創(chuàng)建,用完后不關(guān)閉 Statement 對象,而是放置于 Map 內(nèi)供下一次使用。簡言之,就是重復(fù)使用 Statement 對象;
- BatchExecutor:執(zhí)行 update(沒有 select,jdbc 批處理不支持 select),將所有 SQL 都添加到批處理中(addBatch()),等待統(tǒng)一執(zhí)行(executeBatch()),它緩存了多個 Statement 對象,每個 Statement 對象都是 addBatch()完畢后,等待逐一執(zhí)行 executeBatch()批處理,與 jdbc 批處理相同
7. MyBatis 分頁插件的實(shí)現(xiàn)原理是什么?
分頁插件的基本原理是使用 MyBatis 提供的插件接口,實(shí)現(xiàn)自定義插件,在插件的攔截方法內(nèi)攔截待執(zhí)行的 SQL,然后重寫 SQL,根據(jù) dialect 方言,添加對應(yīng)的物理分頁語句和物理分頁參數(shù)。
8. MyBatis如何返回主鍵?
核心在于 useGeneratedKeys和keyProperty屬性
<mapper namespace="org.chench.test.mybatis.mapper">
<!-- 插入數(shù)據(jù):返回記錄主鍵id值 -->
<insert id="insertOneTest" parameterType="org.chench.test.mybatis.model.Test" useGeneratedKeys="true" keyProperty="id" >
insert into test(name,descr,url,create_time,update_time)
values(#{name},#{descr},#{url},now(),now())
</insert>
</mapper>
9. Xml映射文件中,除了常見的select|insert|update|delete標(biāo)簽之外,還有哪些標(biāo)簽?
還有很多其他的標(biāo)簽例如 sql|cach等...,加上動態(tài)sql的9個標(biāo)簽,trim|where|set|foreach|if|choose|when|otherwise|bind等,其中為sql片段標(biāo)簽,通過標(biāo)簽引入sql片段,為不支持自增的主鍵生成策略標(biāo)簽。
10. MyBatis 和 Hibernate 的區(qū)別有哪些?
- 靈活性:MyBatis 更加靈活,自己可以寫 SQL 語句,使用起來比較方便。
- 可移植性:MyBatis 有很多自己寫的 SQL,因?yàn)槊總€數(shù)據(jù)庫的 SQL 可以不相同,所以可移植性比較差。
- 學(xué)習(xí)和使用門檻:MyBatis 入門比較簡單,使用門檻也更低。
- 二級緩存:hibernate 擁有更好的二級緩存,它的二級緩存可以自行更換為第三方的二級緩存。
總結(jié)
本篇文章就到這里了,希望能給你帶來幫助,也希望你能給多多關(guān)注腳本之家的更多內(nèi)容!
相關(guān)文章
Maven項(xiàng)目分析剔除無用jar引用的方法步驟
這篇文章主要介紹了Maven項(xiàng)目分析剔除無用jar引用的方法步驟,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-10-10
Java使用FastExcel實(shí)現(xiàn)合并單元格
FastExcel 是一個采用純 java 開發(fā)的 excel 文件讀寫組件,支持 Excel'97(-2003)(BIFF8)文件格式,本文主要介紹了如何使用FastExcel實(shí)現(xiàn)合并單元格,需要的可以參考下2024-12-12
Java實(shí)現(xiàn)訂單超時未支付自動取消的8種方法總結(jié)
這篇文章主要為大家介紹了Java實(shí)現(xiàn)訂單超時未支付自動取消功能的8種不同方法,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以了解一下2022-08-08
關(guān)于Java利用反射實(shí)現(xiàn)動態(tài)運(yùn)行一行或多行代碼
這篇文章主要介紹了關(guān)于Java利用反射實(shí)現(xiàn)動態(tài)運(yùn)行一行或多行代碼,借鑒了別人的方法和書上的內(nèi)容,最后將題目完成了,和大家一起分享以下解決方法,需要的朋友可以參考下2023-04-04
Java數(shù)據(jù)結(jié)構(gòu)及算法實(shí)例:選擇排序 Selection Sort
這篇文章主要介紹了Java數(shù)據(jù)結(jié)構(gòu)及算法實(shí)例:選擇排序 Selection Sort,本文直接給出實(shí)現(xiàn)代碼,代碼中包含詳細(xì)注釋,需要的朋友可以參考下2015-06-06

