Mybatis延迟加载和一级缓存之间的问题

本文探讨了Mybatis中延迟加载可能导致的循环问题及解决方案。当使用延迟加载查询学生对象并访问其成绩集时,由于一级缓存的存在,可能会导致查询异常。通过在查询学生语句上设置flushCache='true',可避免因缓存导致的问题。另外,当Spring管理事务时,将@Transactional注解放在service层会导致延迟加载异常,而放在dao层则不会,原因是不同事务层级下SqlSession的关闭时机不同,影响了延迟加载行为。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

问题:在mybatis里面,我们可以使用延迟加载来提高性能,同时减少代码量(只用写查询延时属性的statement语句的id可以),但是这有可能造成循环。比如学生(对应实体类Student)和成绩(对应实体类Score),一个学生有多个成绩,那么在学生实体类Student里面有一个scoreSet的集合,然后在成绩实体类Score里面有一个学生对象Student。

学生类:
public class Student {
	//省略了其他属性和方法
    private String id;
    private String name;
    private Set<Score> scoreSet=new HashSet<Score>();
	}
成绩类:
public class Score {
	//省略了其他属性和方法
     private Integer id;//id,唯一标识
     private Float score1;//一考成绩
     private Student student;//学生类
}

然后在StudentMapper.xml里面写一个StudentResultMap

<resultMap type="com.zxs666.model.Student" id="studentResultMap">
		<id column="id" property="id"/>
		<result column="name" property="name"/>
		<!-- 延迟加载scoreSet  这个select属性就是延迟该学生对应的scoreSet的statementId-->
		<collection property="scoreSet" ofType="com.zxs666.model.Score" column="id"         
                    select="com.zxs666.mapper.ScoreMapper.queryByStudentId" />
	</resultMap>

在ScoreMapper.xml里面写一个ScoreResultMap

<resultMap type="com.zxs666.model.Score" id="scoreResultMap">
  		<id column="id" property="id"/>
  		<result column="score1" property="score1"/>

        <!--延迟加载student  这个select属性就是延迟加载student属性的statementId-->
  		<association property="student" column="studentId"         
          javaType="com.zxs666.model.Student" 
          select="com.zxs666.mapper.StudentMapper.queryById"  ></association>
   </resultMap>

贴上两个查询语句的Id:

StudentMapper.xml根据id唯一查询学生:
    <select id="queryById" parameterType="string" resultMap="studentResultMap">

ScoreMapper.xml根据学生的id查询他的所有成绩:
	<select id="queryByStudentId" parameterType="string" resultMap="scoreResultMap">

现在有一个情况:假设我先查询一个学生(第一次查询),通过StudentMapper.xml里面的queryById这个statement查询,这时候到数据库查询出来的数据,里面的scoreSet是没有值的,当我需要用到scoreSet的时候才会延迟去加载它,问题来了:如果我延迟加载了scoreSet(第二次查询),然后我再去访问scoreSet里面的每一个元素(Score类对象)的student属性,那么这时候将会调用scoreResultMap里面的select属性作为statementId去查询(第三次查询),注意:这里第一次和第三次查询语句的statementId是相同的,所以其实查询的是同一个对象,这个时候mybatis的一级缓存就会返回第一次查询的那个学生对象,那么这样就出现了一个有点像循环的东西,我查询一个学生,然后通过他的属性延迟加载去查询又查到了他本身,这时候就抛出了了一个异常

Exception in thread "main" java.lang.ClassCastException: org.apache.ibatis.executor.ExecutionPlaceholder cannot be cast to java.util.List
	at org.apache.ibatis.executor.BaseExecutor.query(BaseExecutor.java:152)
	at org.apache.ibatis.executor.CachingExecutor.query(CachingExecutor.java:109)
	at org.apache.ibatis.executor.loader.ResultLoader.selectList(ResultLoader.java:81)
	at org.apache.ibatis.executor.loader.ResultLoader.loadResult(ResultLoader.java:70)
	at org.apache.ibatis.executor.loader.ResultLoaderMap$LoadPair.load(ResultLoaderMap.java:219)
	at org.apache.ibatis.executor.loader.ResultLoaderMap$LoadPair.load(ResultLoaderMap.java:184)
	at org.apache.ibatis.executor.loader.ResultLoaderMap.load(ResultLoaderMap.java:81)
	at org.apache.ibatis.executor.loader.ResultLoaderMap.loadAll(ResultLoaderMap.java:95)
	at org.apache.ibatis.executor.loader.javassist.JavassistProxyFactory$EnhancedResultObjectProxyImpl.invoke(JavassistProxyFactory.java:151)

解决办法:在查询学生的那个statement上加一个属性,flushCache=“true”,执行之后,刷新缓存,这样第三次查询的时候缓存里面没有数据,就会重新去数据库查询一个新的student对象。

<select id="queryById" parameterType="string" resultMap="studentResultMap" flushCache="true">

延伸:注意一级缓存的情况,在同一个SqlSession才会去缓存中找数据。我之前的时候出现了一个问题,我使用spring管理事务,当我把事务管理的注解@Transactional加在service层中,并且在service层中做延迟加载,这时候会抛异常,但是如果我把@Transactional加在dao层中时,则不会。分析一下原因:当我把事务控制在service层时,SqlSession在service层不会关闭,在service层延迟加载对数据封装就会出现问题,如果把事务控制在dao层,当从dao查询到数据返回到service层时,sqlSession会关闭,在service层作延迟加载时,不用之前的sqlSession,也就不会出现错误。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值