在项目中大多数用的都是@MapperScan注解,指定basePackages,扫描mybatis Mapper接口类,另外一种方式是用@Mapper注解,其实这两种方法扫描配置用的是一个地方,只是扫描入口不同。
@MapperScan是根据其注解上MapperScannerRegistrar进行自动配置的,最终调用的自动配置代码和下面的代码一致
@Mapper自动配置的程序入口是 MybatisAutoConfiguration类的最下面,位置在mybatis-spring-boot-starter包下面的mybatis-spring-boot-autoconfigure,根据名字可以看出,这个是自动配置,这个地方用到了spring-boot的自动配置相关注解
上面代码的逻辑是 如果标注了@MapperScan 的注解,将会生成 MapperFactoryBean, 如果没有标注@MapperScan 也就是没有MapperFactoryBean的实例,就走@Import里面的配置,下面看看AutoConfiguredMapperScannerRegistrar的配置,它是MybatisAutoConfiguration类下的内部类
上面的代码的逻辑就是初始化ClassPathMapperScanner扫描器,这个扫描器继承了spirng的ClassPathBeanDefinitionScanner的主要作用就是扫描Mapper接口类进行配置并注册为spring bean,首先看看核心的doScan()方法,@MapperScan注解的自动配置也是用了这个注解,这个方法做了什么,也是贴图如下:
这个类主要就是调用父类ClassPathBeanDefinitionScanner中的doScan()方法,获取所有的类定义,之后添加自己的逻辑:processBeanDefinitions()
下面看看processBeanDefinitions()方法主要做了什么,也是贴出核心代码
注意上面涂黄的代码,这个方法主要做的事情就是重新设置上面扫描出的BeanDefinition,设置构造器的参数,构造器的参数为标注了@Mapper的类的class,
做的事情其实是重写BeanDefinition的BeanClass字段为MapperFactoryBean.class,并且将beanClass其实也就是MapperFactoryBean的构造器参数设置为实际的标注了@Mapper的接口,之后当Mapper接口注入的时候,实际调用的是MapperFactoryBean中的getObject()获取特定的mapper实例
如下是MapperFactoryBean的核心逻辑,mapperInterface字段是通过上面的
definition.getConstructorArgumentValues().addGenericArgumentValue(definition.getBeanClassName());这段代码注入的
可见这个类实现了FactoryBean的接口(自由拓展注入的类),以及继承了SqlSessionDaoSupport,继承SqlSessionDaoSupport的主要目的是为了获取SqlSession,通过SqlSession获取具体的mapper代理类(这个也是核心方法),
综上所述,首先根据标注的@MapperScan 获取basePackage或者根据@Mapper获取所在packages,之后通过 ClassPathMapperScanner去扫描包,获取所有Mapper接口类的BeanDefinition,之后具体配置,设置beanClass为MapperFactoryBean,设置MapperFactoryBean的构造器参数为实际的Mapper接口类,通过ClassPathBeanDefinitionScanner父类进行bean注册,自动注入的时候,就会调用MapperFactoryBean的getObject方法获取实际类型的实例。