mybatis 延遲加載的深入理解
什么是延遲加載
延遲加載又叫懶加載,也叫按需加載,也就是說先加載主信息,需要的時(shí)候,再去加載從信息。代碼中有查詢語句,當(dāng)執(zhí)行到查詢語句時(shí),并不是馬上去DB中查詢,而是根據(jù)設(shè)置的延遲策略將查詢向后推遲。
什么時(shí)候會(huì)執(zhí)行延遲加載
配置之后在對(duì)關(guān)聯(lián)對(duì)象進(jìn)行查詢時(shí)使用延遲加載。
延遲加載策略
直接加載
遇到代碼中查詢語句,馬上到DB中執(zhí)行select語句進(jìn)行查詢。(這種只能用于多表單獨(dú)查詢)
侵入式延遲加載
將關(guān)聯(lián)對(duì)象的詳情(具體數(shù)據(jù),如id、name)侵入到主加載對(duì)象,作為主加載對(duì)象的詳情的一部分出現(xiàn)。當(dāng)要訪問主加載對(duì)象的詳情時(shí)才會(huì)查詢主表,但由于關(guān)聯(lián)對(duì)象詳情作為主加載對(duì)象的詳情一部分出現(xiàn),所以這個(gè)查詢不僅會(huì)查詢主表,還會(huì)查詢關(guān)聯(lián)表。
深度延遲加載
將關(guān)聯(lián)對(duì)象的詳情(具體數(shù)據(jù),如id、name)侵入到主加載對(duì)象,作為主加載對(duì)象的詳情的一部分出現(xiàn)。當(dāng)要訪問主加載對(duì)象的詳情時(shí)才會(huì)查詢主表,但由于關(guān)聯(lián)對(duì)象詳情作為主加載對(duì)象的詳情一部分出現(xiàn),所以這個(gè)查詢不僅會(huì)查詢主表,還會(huì)查詢關(guān)聯(lián)表。
使用延遲加載的目的
減輕DB服務(wù)器的壓力,因?yàn)槲覀冄舆t加載只有在用到需要的數(shù)據(jù)才會(huì)執(zhí)行查詢操作。
如何開啟延遲加載功能
Mybatis的延遲加載功能默認(rèn)是關(guān)閉的
需要在SqlMapConfig.xml文件中通過setting標(biāo)簽配置來開啟延遲加載功能
開啟延遲加載的屬性:
- lazyLoadingEnabled:全局性設(shè)置懶加載。如果設(shè)為‘false',則所有相關(guān)聯(lián)的都會(huì)被初始化加載。默認(rèn)為false
- aggressiveLazyLoading:當(dāng)設(shè)置為‘true'的時(shí)候,懶加載的對(duì)象可能被任何懶屬性全部加載。否則,每個(gè)屬性都按需加載。默認(rèn)為true
配置
<settings>
<setting name ="aggressiveLazyLoading" value="false"/>
<!--開啟延遲加載-->
<setting name="lazyLoadingEnabled" value="true"/>
</settings>
我們用關(guān)聯(lián)查詢來實(shí)現(xiàn)延遲加載,假設(shè)我們現(xiàn)在要查出用戶和用戶角色。
首先我們?cè)趗ser中添加查詢userVo的方法和xml。
<!--userMapper.xml-->
....
<resultMap id="BaseResultMap" type="com.redstar.basemapper.pojo.User">
<id column="id" jdbcType="VARCHAR" property="id"/>
<result column="name" jdbcType="VARCHAR" property="name"/>
<result column="age" jdbcType="INTEGER" property="age"/>
<result column="role_id" jdbcType="INTEGER" property="roleId"/>
</resultMap>
<resultMap id="userRoleMapSelect" type="com.redstar.basemapper.pojo.UserVo">
<association property="user" resultMap="BaseResultMap"/>
<association property="role" fetchType="lazy" column="{id=role_id}"
select="com.redstar.basemapper.dao.RoleMapper.getRoleById"/>
</resultMap>
<sql id="Base_Column_List">
id, `name`, age, role_id
</sql>
<select id="getUserVo" resultMap="userRoleMapSelect">
select * from user where id=#{userId}
</select>
...
<!--roleMapper.xml-->
...
<resultMap id="BaseResultMap" type="com.redstar.basemapper.pojo.Role">
<id column="id" jdbcType="INTEGER" property="id" />
<result column="role_name" jdbcType="VARCHAR" property="roleName" />
</resultMap>
<sql id="Base_Column_List">
id, role_name
</sql>
<select id="getRoleById" resultMap="BaseResultMap">
select * from role where id=#{id}
</select>
...
測試用例
@RunWith(SpringRunner.class)
@SpringBootTest
public class BaseMapperApplicationTests {
@Autowired
private UserMapper userMapper;
@Autowired
private RoleMapper roleMapper;
@Test
public void getUserVo() {
System.out.println(userMapper.getUserVo("12312232"));
// System.out.println(userMapper.getUserById("12312232"));
// System.out.println(roleMapper.getRoleById(1));
}
}
輸出結(jié)果:
UserVo{user=User [Hash = 1937575946, id=12312232, name=哇哈哈啊娃哈哈哇哈哈哈哈哈哈哈, age=48, roleId=1, serialVersionUID=1], role=Role [Hash = 317053574, id=1, roleName=admin, serialVersionUID=1]}
注意
許多對(duì)延遲加載原理不太熟悉的朋友會(huì)經(jīng)常遇到一些莫名其妙的問題:有些時(shí)候延遲加載可以得到數(shù)據(jù),有些時(shí)候延遲加載就會(huì)報(bào)錯(cuò),為什么會(huì)出現(xiàn)這種情況呢?
MyBatis 延遲加載是通過動(dòng)態(tài)代理實(shí)現(xiàn)的,當(dāng)調(diào)用配直為延遲加載的屬性方法時(shí), 動(dòng)態(tài)代理的操作會(huì)被觸發(fā),這些額外的操作就是通過 MyBatis 的 SqlSessio口去執(zhí)行嵌套 SQL 的 。由于在和某些框架集成時(shí), SqlSession 的生命周期交給了框架來管理,因此當(dāng)對(duì)象超出SqlSession 生命周期調(diào)用時(shí),會(huì)由于鏈接關(guān)閉等問題而拋出異常 。 在和 Spring 集成時(shí),要確保只能在 Service 層調(diào)用延遲加載的屬性 。 當(dāng)結(jié)果從 Service 層返回至 Controller 層時(shí), 如果獲取延遲加載的屬性值,會(huì)因?yàn)?SqlSessio口已經(jīng)關(guān)閉而拋出異常 。
以上就是本文的全部內(nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Java springboot 配置文件與多環(huán)境配置與運(yùn)行優(yōu)先級(jí)
這篇文章主要介紹了Java springboot如何配置文件,進(jìn)行多環(huán)境配置,以及運(yùn)行優(yōu)先級(jí),感興趣的小伙伴可以借鑒一下2023-04-04
Spring Cloud Config Client超時(shí)及重試示例詳解
這篇文章主要給大家介紹了關(guān)于Spring Cloud Config Client超時(shí)及重試的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧2018-05-05
Java?C++算法題解leetcode801使序列遞增的最小交換次數(shù)
這篇文章主要為大家介紹了Java?C++題解leetcode801使序列遞增的最小交換次數(shù)示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-10-10
redis發(fā)布訂閱Java代碼實(shí)現(xiàn)過程解析
這篇文章主要介紹了redis發(fā)布訂閱Java代碼實(shí)現(xiàn)過程解析,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-09-09
Java反射之類的實(shí)例對(duì)象的三種表示方式總結(jié)
下面小編就為大家?guī)硪黄狫ava反射之類的實(shí)例對(duì)象的三種表示方式總結(jié)。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2016-10-10
java 日期各種格式之間的相互轉(zhuǎn)換實(shí)例代碼
這篇文章主要介紹了java 日期各種格式之間的相互轉(zhuǎn)換實(shí)例代碼的相關(guān)資料,需要的朋友可以參考下2017-02-02
Netty結(jié)合Protobuf進(jìn)行編解碼的方法
這篇文章主要介紹了Netty結(jié)合Protobuf進(jìn)行編解碼,通過文檔表述和代碼實(shí)例充分說明了如何進(jìn)行使用和操作,需要的朋友可以參考下2021-06-06
使用feign傳遞參數(shù)類型為MultipartFile的問題
這篇文章主要介紹了使用feign傳遞參數(shù)類型為MultipartFile的問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-03-03
Java如何使用流去除集合中某個(gè)字段為空的對(duì)象
這篇文章主要給大家介紹了關(guān)于Java如何使用流去除集合中某個(gè)字段為空的對(duì)象,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用Java具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2023-08-08

