mybatis缓存-延迟加载

设置延迟加载

什么是延迟加载

resultMap可实现高级映射(使用association、collection实现一对一及一对多映射),association、collection具备延迟加载功能。

​ 延迟加载的意思是说,在关联查询时,利用延迟加载,先加载主信息。需要关联信息时再去按需加载关联信息。这样会大大提高数据库性能,因为查询单表要比关联查询多张表速度要快。减少内存的开销。

导入 cglib-nodep.jar

<dependency>
    <groupId>cglibgroupId>
    <artifactId>cglib-nodepartifactId>
    <version>3.1version>
dependency>

设置延迟加载

Mybatis默认是不开启延迟加载功能的,我们需要手动开启。

需要在SqlMapConfig.xml文件中,在标签中开启延迟加载功能。

lazyLoadingEnabledaggressiveLazyLoading

设置项 描述 允许值 默认值
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>

延迟加载案例association或collection

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(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)]

你可能感兴趣的:(mybatis,java,数据库,mybatis)