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