1.@MapperScan如何让Spring知道Mapper的定义
@MapperScan中的注解@Import(MapperScannerRegistrar.class)引入了public class MapperScannerRegistrar implements ImportBeanDefinitionRegistrar, ResourceLoaderAware {}
MapperScannerRegistrar重写了ImportBeanDefinitionRegistrar的 public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry)根据包名扫描所有的class,生成Mapper的beanDefinitions。
在processBeanDefinitions(Set beanDefinitions)时将Mapper的Mapper的BeanDefinition的class替换为mapperFactoryBean.getClass() 、beanDefiniton将自己的构造器参数设置为mapper的className 、 设置beanDefinition sqlSessionFactory属性的RuntimeBeanReference (ref)(在Spring处理依赖关系时,最终会将该引用替换成实际生成的Bean对象)
2.如何执行SQL
每个MapperFactoryBean实例化时,就构造注入了mapperInterface的class,从SqlSession获取Mapper的时候用到。MapperFactoryBean创建Mapper代理对象,在Mapper代理对象中执行sqlSession的执行方法。其实这个sqlSession是一个实现了SqlSession接口的SqlSessionTemplate,在MapperFactoryBean创建后被装配。
在SqlSessionTemplate中,有一个叫sqlSessionProxy的代理sqlSession,在创建SqlSessionTemplate的构造方法中被反射创建,反射处理器为SqlSessionInterceptor。
sqlSessionTemplate是对sqlSessionProxy的包装,增删改查调用的是sqlSessionProxy里的方法,所以最终执行的是SqlSessionInterceptor这个反射处理器的invoke方法。
invoke通过getSqlSession方法拿到事务上下文DefaultSqlSession实例--getSqlSession方法里会从事务管理器中通过ThreadLocal获取 SqlSessionHolder(sqlSession的包装),进而获取SqlSession,如果获取不到则从SqlSessionFactory获取并注册到事务管理器中。
实际和数据库打交道的还是DefaultSqlSession