Mybatis和Spring是如何整合的?

需要了解的前置知识:

  • Bean的生命周期
  • JDK动态代理
  • FactoryBean实例化

在spring的配置文件中引入下面两个bean,spring启动的时候,会加载这两个bean

<bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
    <property name="dataSource" ref="dataSource"/>
    <property name="configLocation" value="classpath:mybatis-config.xml"/>
    <property name="mapperLocations" value="classpath:mapper/*.xml"/>
    <property name="typeAliasesPackage" value="com.tt.spring.demo.pojo"/>
</bean>

<bean class="org.mybatis.spring.mapper.MapperScannerConfigurer">
    <property name="sqlSessionFactoryBeanName" value="sqlSessionFactory"/>
    <property name="basePackage" value="com.tt.spring.demo.dao"/>
</bean>

MapperScannerConfigurer实现了BeanDefinitionRegistryPostProcessor,因此spring会先进行实例化,并调用postProcessBeanDefinitionRegistry方法

在MapperScannerConfigurer类的postProcessBeanDefinitionRegistry方法中,创建了ClassPathMapperScanner对象,并调用了scan方法进行bean扫描

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

    ClassPathMapperScanner scanner = new ClassPathMapperScanner(registry);

	// ...省略一堆set方法
    
    scanner.scan(
        StringUtils.tokenizeToStringArray(this.basePackage, ConfigurableApplicationContext.CONFIG_LOCATION_DELIMITERS));
  }

在ClassPathMapperScanner的doScan方法中,根据指定的basePackages,扫描指定的bean(通过重写isCandidateComponent方法,让接口可以作为beanDefinition)

  protected boolean isCandidateComponent(AnnotatedBeanDefinition beanDefinition) {
    return beanDefinition.getMetadata().isInterface() && beanDefinition.getMetadata().isIndependent();
  }

扫描完所有的beanDefinition后,调用processBeanDefinitions方法对beanDefinition进行定制,我们重点看两个地方:

  private void processBeanDefinitions(Set<BeanDefinitionHolder> beanDefinitions) {
      for (BeanDefinitionHolder holder : beanDefinitions) {
        definition.setBeanClass(this.mapperFactoryBeanClass);	// (1)
        
      if (StringUtils.hasText(this.sqlSessionFactoryBeanName)) {
          // 我们在上面定义了sqlSessionFactoryBeanName,因此会走这个分支
        definition.getPropertyValues().add("sqlSessionFactory",
            new RuntimeBeanReference(this.sqlSessionFactoryBeanName)); // (2)
        explicitFactoryUsed = true;
      } else if (this.sqlSessionFactory != null) {
        definition.getPropertyValues().add("sqlSessionFactory", this.sqlSessionFactory);
        explicitFactoryUsed = true;
      }
  }
  • 代码(1)处,创建的实例实际就是mapperFactoryBeanClass对应的实例
  • 代码(2)处,创建实例后会先创建sqlSessionFactoryBeanName对应的实例,然后调用setSqlSessionFactory方法

最后将这些beanDefinitions都注册到spring中

实例化:

实例化会创建MapperFactoryBean对象,它是一个FactoryBean,getObject方法的返回值才是真正的bean对象

  public void setSqlSessionFactory(SqlSessionFactory sqlSessionFactory) {
    if (this.sqlSessionTemplate == null || sqlSessionFactory != this.sqlSessionTemplate.getSqlSessionFactory()) {
      this.sqlSessionTemplate = createSqlSessionTemplate(sqlSessionFactory);
    }
  }
  • 5
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值