MyBatis关键配置-接口的扫描注册
在Service 層可以使用@Autowired 自動注入的Mapper 接口, 需要保存在BeanFactory(比如XmlWebApplicationContext)中。也就是說接口肯定是在Spring啟動的時候被掃描了,注冊過的。
1、什么時候掃描的?
2、注冊的時候,注冊的是什么?這個決定了我們拿到的是什么實際對象。
回顧一下, 我們在applicationContext.xml 里面配置了一個MapperScannerConfigurer。
MapperScannerConfigurer 實現了BeanDefinitionRegistryPostProcessor 接口,BeanDefinitionRegistryPostProcessor 是BeanFactoryPostProcessor 的子類,可以通過編碼的方式修改、新增或者刪除某些Bean 的定義。
?我們只需要重寫postProcessBeanDefinitionRegistry()方法,在這里面操作Bean就可以了。
在這個方法里面:
scanner.scan() 方法是ClassPathBeanDefinitionScanner 中的, 而它的子類ClassPathMapperScanner 覆蓋了doScan() 方法, 在doScan() 中調用了processBeanDefinitions:
public Set<BeanDefinitionHolder> doScan(String... basePackages) {Set<BeanDefinitionHolder> beanDefinitions = super.doScan(basePackages);if (beanDefinitions.isEmpty()) {LOGGER.warn(() -> "No MyBatis mapper was found in '" + Arrays.toString(basePackages) +"' package. Please check your configuration.");} else {processBeanDefinitions(beanDefinitions);}return beanDefinitions; }它先調用父類的doScan()掃描所有的接口。
processBeanDefinitions 方法里面,在注冊beanDefinitions 的時候,BeanClass被改為MapperFactoryBean(注意灰色的注釋)。
private void processBeanDefinitions(Set<BeanDefinitionHolder> beanDefinitions) {GenericBeanDefinition definition;for (BeanDefinitionHolder holder : beanDefinitions) {definition = (GenericBeanDefinition) holder.getBeanDefinition();String beanClassName = definition.getBeanClassName();LOGGER.debug(() -> "Creating MapperFactoryBean with name '" + holder.getBeanName()+ "' and '" + beanClassName + "' mapperInterface");// the mapper interface is the original class of the bean// but, the actual class of the bean is MapperFactoryBeandefinition.getConstructorArgumentValues().addGenericArgumentValue(beanClassName); //issue #59definition.setBeanClass(this.mapperFactoryBean.getClass());問題:
為什么要把BeanClass 修改成MapperFactoryBean,這個類有什么作用?MapperFactoryBean 繼承了SqlSessionDaoSupport , 可以拿到SqlSessionTemplate。
?
總結
以上是生活随笔為你收集整理的MyBatis关键配置-接口的扫描注册的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: MyBatis创建SqlSession-
- 下一篇: MyBatis关键配置-接口注入使用