Mybatis源码分析——如何为dao类生成代理对象

本文详细分析了Mybatis如何在没有DAO实现类的情况下,通过动态代理生成DAO接口的代理对象。从MapperScannerConfigurer的扫描配置,到MapperFactoryBean的设置,再到MapperProxy的创建,揭示了Mybatis在运行时如何利用SqlSession执行SQL的过程。
摘要由CSDN通过智能技术生成

Mybatis入门,一个简单的demo

在上面demo中,我们定义了一个dao接口,并没有实现类,然后直接调用接口的方法,就可以执行mapper里的sql。这是怎么做到的呢。

Mybatis源码分析——mapper.xml解析

在这篇文章中,我们分析到:mapper.xml对应的dao类会添加到configuration的MapperRegistry的Map<Class<?>, MapperProxyFactory<?>> knownMappers里,key是类的全名,value是对应的代理对象。


入口就在这里,在xml配置文件中,我们会有以下配置,启动扫描配置路径下的dao类

<!-- 自动扫描了所有的XxxxMapper.xml对应的mapper接口文件,这样就不用一个一个手动配置Mpper的映射了,只要Mapper接口类和Mapper映射文件对应起来就可以了。 -->
<bean class="org.mybatis.spring.mapper.MapperScannerConfigurer">
	<property name="basePackage" value="com.rice.mvc.dao" />
</bean>

我们来看看MapperScannerConfigurer类,它实现了BeanDefinitionRegistryPostProcessor,我们找到对应的postProcessBeanDefinitionRegistry方法,如下

  @Override
  public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) {
    if (this.processPropertyPlaceHolders) {
      processPropertyPlaceHolders();
    }

    ClassPathMapperScanner scanner = new ClassPathMapperScanner(registry);
    scanner.setAddToConfig(this.addToConfig);
    scanner.setAnnotationClass(this.annotationClass);
    scanner.setMarkerInterface(this.markerInterface);
    scanner.setSqlSessionFactory(this.sqlSessionFactory);
    scanner.setSqlSessionTemplate(this.sqlSessionTemplate);
    scanner.setSqlSessionFactoryBeanName(this.sqlSessionFactoryBeanName);
    scanner.setSqlSessionTemplateBeanName(this.sqlSessionTemplateBeanName);
    scanner.setResourceLoader(this.applicationContext);
    scanner.setBeanNameGenerator(this.nameGenerator);
    scanner.registerFilters();
    scanner.scan(StringUtils.tokenizeToStringArray(this.basePackage, ConfigurableApplicationContext.CONFIG_LOCATION_DELIMITERS));
  }

processPropertyPlaceHolders();方法是做配置解析的。重点在ClassPathMapperScanner.scan方法。ClassPathMapperScanner继承于ClassPathBeanDefinitionScanner。来看scan方法,这是ClassPathBeanDefinitionScanner.scan,如下

public int scan(String... basePackages) {
	int beanCountAtScanStart = this.registry.getBeanDefinitionCount();

	doScan(basePackages);

	// Register annotation config processors, if necessary.
	if (this.includeAnnotationConfig) {
		AnnotationConfigUtils.registerAnnotationConfigProcessors(this.registry);
	}

	return (this.registry.getBeanDefinitionCount() - beanCountAtScanStart);
}

再看doScan,这里调用的是ClassPathMapperScanner.doScan,如下

  @Override
  public Set<BeanDefinitionHolder> doScan(String... basePackages) {
    Set<BeanDefinitionHolder> beanDefinitions = super.doScan(basePackages);

    if (
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值