核心代码
1.解析配置类上的@MapperScan(“com.liqi.mapper”)
@Import(MapperScannerRegistrar.class)
会调用MapperScannerRegistrar.registerBeanDefinitions
2. ClassPathMapperScanner scanner = new ClassPathMapperScanner(registry);
scanner.doScan(StringUtils.toStringArray(basePackages));
拿到路径,扫描“com.liqi.mapper”路径
3.重新doScan方法,改变beanDefinition的beanClass属性
definition.setBeanClass(this.mapperFactoryBean.getClass());
4.MapperFactoryBean.getObject,获取mapper接口的代理对象返回。
以上就是整合mybatis的核心步骤。
spring整合mybaits一级缓存失效的问题
1.mybatis源码中将sqlSession放到了threadLocal中,执行sql,其实调用的是sqlSession.SelectOne()方法,这样同一是执行命令,直接从sqlSession中返回。
2.由于mybatis的sqlSession底层中没有加锁,线程是不安全的,spring整合mybatis的时候,获取sqlSession并发放到threadLocal中,所以每一个执行命令,都会获取不同的sqlSession,解决了安全问题,但同时一级缓存也失效了。
3.如何解决失效问题:加@Transactional注解,spring底层有判断,如果有这个注解,就会放到threadLocal里,但一般我们不用mybatis的一级缓存,例如我们的数据库隔离级别是读未提交,如果有一级缓存,数据库的隔离级别失效。
以上是我对spring整合mybatis的理解。