resultMap可实现高级映射(使用association、collection实现一对一及一对多映射),association、collection具备延迟加载功能。
延迟加载的意思是说,在关联查询时,利用延迟加载,先加载主信息。需要关联信息时再去按需加载关联信息。这样会大大提高数据库性能,因为查询单表要比关联查询多张表速度要快。减少内存的开销。
导入 cglib-nodep.jar
<dependency>
<groupId>cglibgroupId>
<artifactId>cglib-nodepartifactId>
<version>3.1version>
dependency>
Mybatis默认是不开启延迟加载功能的,我们需要手动开启。
需要在SqlMapConfig.xml文件中,在标签中开启延迟加载功能。
lazyLoadingEnabled、aggressiveLazyLoading
设置项 | 描述 | 允许值 | 默认值 |
---|---|---|---|
lazyLoadingEnabled | 全局性设置懒加载。如果设为‘false’,则所有相关联的都会被初始化加载。 | true | false | false |
aggressiveLazyLoading | 当设置为‘true’的时候,懒加载的对象可能被任何懒属性全部加载。否则,每个属性都按需加载。 | true | false | true |
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-OJdFEIUs-1618547259982)(http://m.qpic.cn/psb?/V13x1ZYF1dFQtq/CSRsiHleyfyRQIjdHe*rGcsUpeCTEhbPgVdggBq5s7E!/b/dL8AAAAAAAAA&bo=5AJ1AQAAAAADB7A!&rf=viewer_4)]
<settings>
<setting name="lazyLoadingEnabled" value="true" />
<setting name="aggressiveLazyLoading" value="false"/>
settings>
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-JIlYe7p9-1618547259985)(http://m.qpic.cn/psb?/V13x1ZYF1dFQtq/wZtD3Q6o1goe1X9AP5Bhe7Y3wAZ7J2XadITZfcvQpM0!/b/dLYAAAAAAAAA&bo=YQQ9AQAAAAADB3s!&rf=viewer_4)]
执行结果很明显,使用了延迟加载,将关联查询分成了两次单表查询。
观察:如果同样的功能查询两次,就是第二次查用户的时候,看有没有什么变化?
mybatis提供查询缓存,如果缓存中有数据就不用从数据库中获取,用于减轻数据压力,提高系统性能。
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-6yV1X1V5-1618547259988)(http://m.qpic.cn/psb?/V13x1ZYF1dFQtq/BdFGx2fYZUooDqfwqLcU6opbuqMjeqGhJm13lmZ*DzA!/b/dLgAAAAAAAAA&bo=NQNoAQAAAAADB30!&rf=viewer_4)]
一级缓存是SqlSession级别的缓存。在操作数据库时需要构造 sqlSession对象,在对象中有一个数据结构(HashMap)用于存储缓存数据。不同的sqlSession之间的缓存数据区域(HashMap)是互相不影响的。
二级缓存是mapper级别的缓存,多个SqlSession去操作同一个Mapper的sql语句,多个SqlSession可以共用二级缓存,二级缓存是跨SqlSession的。
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-t7A5G5Bz-1618547259991)(http://m.qpic.cn/psb?/V13x1ZYF1dFQtq/oqT3*BUQdsnFON3EFjW7SeuRaH8LTx9p**lxRKX7lLY!/b/dDUBAAAAAAAA&bo=wQKzAQAAAAADB1M!&rf=viewer_4)]
第一次发起查询用户id为1的用户信息,先去找缓存中是否有id为1的用户信息,如果没有,从数据库查询用户信息。
得到用户信息,将用户信息存储到一级缓存中。
如果sqlSession去执行commit操作(执行插入、更新、删除),清空SqlSession中的一级缓存,这样做的目的为了让缓存中存储的是最新的信息,避免脏读。
第二次发起查询用户id为1的用户信息,先去找缓存中是否有id为1的用户信息,缓存中有,直接从缓存中获取用户信息。
Mybatis默认支持一级缓存。
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-o7oD7DZC-1618547259995)(http://m.qpic.cn/psb?/V13x1ZYF1dFQtq/2NgnXGCzzq7*rOC3CPdzs2lxBIT9hMznXWJitls77y0!/b/dLYAAAAAAAAA&bo=CgMxAQAAAAADBxs!&rf=viewer_4)]
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-7nAa3Bve-1618547259997)(http://m.qpic.cn/psb?/V13x1ZYF1dFQtq/BzNrkm12P767nGE6XS98NOHWCTSWwhqDm2Ayk2xKE!/b/dDcBAAAAAAAA&bo=BgQwAQAAAAADFwE!&rf=viewer_4)]
session方法结束,sqlSession就关闭,一级缓存就清空。
下图是多个sqlSession请求UserMapper的二级缓存图解。
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-gmRpAKm9-1618547259998)(http://m.qpic.cn/psb?/V13x1ZYF1dFQtq/qupQr6z2J1LzyTBJl8jHlO0MRVC55hAKIYfWPFzpQ!/b/dFMBAAAAAAAA&bo=2QKzAQAAAAADB0s!&rf=viewer_4)]
二级缓存是mapper级别的。
第一次调用mapper下的SQL去查询用户信息。查询到的信息会存到该mapper对应的二级缓存区域内。
第二次调用相同namespace下的mapper映射文件中相同的SQL去查询用户信息。会去对应的二级缓存内取结果。
如果调用相同namespace下的mapper映射文件中的增删改SQL,并执行了commit操作。此时会清空该namespace下的二级缓存。
Mybatis默认是没有开启二级缓存
1、 在核心配置文件SqlMapConfig.xml(mybatis.xml)中加入以下内容(开启二级缓存总开关):
在settings标签中添加以下内容:
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-cOayVFlI-1618547259999)(http://m.qpic.cn/psb?/V13x1ZYF1dFQtq/LD9bZm0VV3jYRY83EIlD507UVJOnYpzpcpIKyNia3Fg!/b/dL8AAAAAAAAA&bo=lgLLAAAAAAADB30!&rf=viewer_4)]
2、 在对应的 映射文件中,加入以下内容,给其开启二级缓存:
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-03PHCZGU-1618547260000)(http://m.qpic.cn/psb?/V13x1ZYF1dFQtq/UlJQgBXej46uMJ0Ep8QtyHTMpSC4hoyVzEl6GdoIlLQ!/b/dL8AAAAAAAAA&bo=5QGbAAAAAAADB10!&rf=viewer_4)]
3、对类完成序列化操作
由于二级缓存的数据不一定都是存储到内存中,它的存储介质多种多样,所以需要给缓存的对象执行序列化。
如果该类存在父类,那么父类也要实现序列化。
序列化作用: 为了方便网络传输
序列化:将对象转换为字节序列的过程称为序列化
反序列化:将字节序列恢复为对象的过程称为反序列化
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-8Q1R2Krl-1618547260001)(http://m.qpic.cn/psb?/V13x1ZYF1dFQtq/Ce0wpmhKX2x8JX5UR39aFysa5wSNdMTphMecxN6JTdE!/b/dLYAAAAAAAAA&bo=IgJbAAAAAAADB1k!&rf=viewer_4)]
4、测试
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-V0JjBqaE-1618547260002)(http://m.qpic.cn/psb?/V13x1ZYF1dFQtq/7OfxDOEh3UyxRZFHiR38vwHuMaNaTEmJA*5aki86Vb0!/b/dEYBAAAAAAAA&bo=9gIgAgAAAAADF.Q!&rf=viewer_4)]
该statement中设置useCache=false,可以禁用当前select语句的二级缓存,即每次查询都是去数据库中查询,默认情况下是true,即该statement使用二级缓存
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-v4vio2CN-1618547260003)(http://m.qpic.cn/psb?/V13x1ZYF1dFQtq/paH5g70zAu.Bzush8GjCCPph43yLbzHtWbMCW1SooVQ!/b/dLsAAAAAAAAA&bo=1QJVAAAAAAADB6A!&rf=viewer_4)]
该statement中设置flushCache=true可以刷新当前的二级缓存 默认情况下如果是select语句,那么flushCache是false。如果是insert、update、delete语句,那么flushCache是true。
如果查询语句设置成true,那么每次查询都是去数据库查询,即意味着该查询的二级缓存失效。
如果查询语句设置成false,即使用二级缓存,那么如果在数据库中修改了数据,而缓存数据还是原来的,这个时候就会出现脏读。
flushCache设置如下:
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-H9hBjsNu-1618547260004)(http://m.qpic.cn/psb?/V13x1ZYF1dFQtq/NcKlbvNmLCDmSjN4WreFreA2TEPVb6lLmCBvl6SKWUQ!/b/dLgAAAAAAAAA&bo=3AJfAAAAAAADB6M!&rf=viewer_4)]
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-Qm5fKFYg-1618547260005)(http://m.qpic.cn/psb?/V13x1ZYF1dFQtq/KWTZ441VTVEOfjRLisMC8yg0u*jiFP9iuWjsc8FRUd0!/b/dDUBAAAAAAAA&bo=lgMXAgAAAAADF7I!&rf=viewer_4)]