什么是延遲加載
延遲加載又叫懶加載,也叫按需加載,也就是說先加載主信息,需要的時候,再去加載從信息。代碼中有查詢語句,當(dāng)執(zhí)行到查詢語句時,并不是馬上去db中查詢,而是根據(jù)設(shè)置的延遲策略將查詢向后推遲。
什么時候會執(zhí)行延遲加載
配置之后在對關(guān)聯(lián)對象進行查詢時使用延遲加載。
延遲加載策略
直接加載
遇到代碼中查詢語句,馬上到db中執(zhí)行select語句進行查詢。(這種只能用于多表單獨查詢)
侵入式延遲加載
將關(guān)聯(lián)對象的詳情(具體數(shù)據(jù),如id、name)侵入到主加載對象,作為主加載對象的詳情的一部分出現(xiàn)。當(dāng)要訪問主加載對象的詳情時才會查詢主表,但由于關(guān)聯(lián)對象詳情作為主加載對象的詳情一部分出現(xiàn),所以這個查詢不僅會查詢主表,還會查詢關(guān)聯(lián)表。
深度延遲加載
將關(guān)聯(lián)對象的詳情(具體數(shù)據(jù),如id、name)侵入到主加載對象,作為主加載對象的詳情的一部分出現(xiàn)。當(dāng)要訪問主加載對象的詳情時才會查詢主表,但由于關(guān)聯(lián)對象詳情作為主加載對象的詳情一部分出現(xiàn),所以這個查詢不僅會查詢主表,還會查詢關(guān)聯(lián)表。
使用延遲加載的目的
減輕db服務(wù)器的壓力,因為我們延遲加載只有在用到需要的數(shù)據(jù)才會執(zhí)行查詢操作。
如何開啟延遲加載功能
mybatis的延遲加載功能默認(rèn)是關(guān)閉的
需要在sqlmapconfig.xml文件中通過setting標(biāo)簽配置來開啟延遲加載功能
開啟延遲加載的屬性:
- lazyloadingenabled:全局性設(shè)置懶加載。如果設(shè)為‘false',則所有相關(guān)聯(lián)的都會被初始化加載。默認(rèn)為false
- aggressivelazyloading:當(dāng)設(shè)置為‘true'的時候,懶加載的對象可能被任何懶屬性全部加載。否則,每個屬性都按需加載。默認(rèn)為true
配置
1
2
3
4
5
|
<settings> <setting name = "aggressivelazyloading" value= "false" /> <!--開啟延遲加載--> <setting name= "lazyloadingenabled" value= "true" /> </settings> |
我們用關(guān)聯(lián)查詢來實現(xiàn)延遲加載,假設(shè)我們現(xiàn)在要查出用戶和用戶角色。
首先我們在user中添加查詢uservo的方法和xml。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
|
<!--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> ... |
測試用例
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
@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]}
注意
許多對延遲加載原理不太熟悉的朋友會經(jīng)常遇到一些莫名其妙的問題:有些時候延遲加載可以得到數(shù)據(jù),有些時候延遲加載就會報錯,為什么會出現(xiàn)這種情況呢?
mybatis 延遲加載是通過動態(tài)代理實現(xiàn)的,當(dāng)調(diào)用配直為延遲加載的屬性方法時, 動態(tài)代理的操作會被觸發(fā),這些額外的操作就是通過 mybatis 的 sqlsessio口去執(zhí)行嵌套 sql 的 。由于在和某些框架集成時, sqlsession 的生命周期交給了框架來管理,因此當(dāng)對象超出sqlsession 生命周期調(diào)用時,會由于鏈接關(guān)閉等問題而拋出異常 。 在和 spring 集成時,要確保只能在 service 層調(diào)用延遲加載的屬性 。 當(dāng)結(jié)果從 service 層返回至 controller 層時, 如果獲取延遲加載的屬性值,會因為 sqlsessio口已經(jīng)關(guān)閉而拋出異常 。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持服務(wù)器之家。
原文鏈接:https://segmentfault.com/a/1190000017895924