spring集成mybatis源码跟踪

注:为了便于理解源码逻辑,本文只截取了核心源码。

一、调用图解

二、源码解析

使用spring集成mybatis时只需要简单几步,此文不再赘述,下面我们从我们的配置类作为入口,一步步探索spring集成mybatis的过程:

  1. 首先是我们的配置类,我们的配置类中:
    • 标注@MapperScan注解,值为扫描mapper包
    • 一个SqlSessionFactory的@Bean方法
@Configuration
@ComponentScan("com.lk.demo.mybatis_spring")
@MapperScan("com.lk.demo.mybatis_spring")
public class Config {
    @Bean
    public SqlSessionFactory sqlSessionFactory() throws Exception {
        SqlSessionFactoryBean factoryBean = new SqlSessionFactoryBean();
        factoryBean.setDataSource(dataSource());
        return factoryBean.getObject();
    }

    @Bean
    DataSource dataSource(){
        DruidDataSource dataSource = new DruidDataSource();
        dataSource.setUrl("jdbc:mysql://192.168.1.216:3306/lk_db");
        dataSource.setUsername("root");
        dataSource.setPassword("123");
        return dataSource;
    }
}
  1. 然后我们点进 MapperScan 注解。
    看到了什么,老套路,通过import引入类,我们先别点进去,想想,这个MapperScannerRegistrar里面做了什么呢?想不到是吧,没关系,带着这个问题,下面几步有答案。
@Import(MapperScannerRegistrar.class)
public @interface MapperScan {
}
  1. 继续,进入MapperScannerRegistrar
    它的父类是ImportBeanDefinitionRegistrar,那我们就看registerBeanDefinitions方法
public class MapperScannerRegistrar implements ImportBeanDefinitionRegistrar, ResourceLoaderAware {

    @Override
    public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {

        AnnotationAttributes annoAttrs = AnnotationAttributes.fromMap(importingClassMetadata.getAnnotationAttributes(MapperScan.class.getName()));
        ClassPathMapperScanner scanner = new ClassPathMapperScanner(registry);

        List<String> basePackages = new ArrayList<String>();
        for (String pkg : annoAttrs.getStringArray("value")) {
            if (StringUtils.hasText(pkg)) {
                basePackages.add(pkg);
            }
        }
        scanner.registerFilters();
        scanner.doScan(StringUtils.toStringArray(basePackages));
    }
}

哦~,MapperScannerRegistrar里面创建了一个mapper扫描器(ClassPathMapperScanner),然后用它去扫描,扫描做了什么呢?

  1. 进入ClassPathMapperScanner
    ClassPathMapperScanner有4个重要的方法:

    • registerFilters()
    • doScan()
    • processBeanDefinitions()
    • isCandidateComponent()

    1个重要的属性:

    • mapperFactoryBean
public class ClassPathMapperScanner extends ClassPathBeanDefinitionScanner {
    private MapperFactoryBean<?> mapperFactoryBean = new MapperFactoryBean<Object>();
    
    // spring的scanner都是通过过滤器初步筛选类,所以这里加了一个IncludeFilter,它的作用是初筛剩下所有的类
    public void registerFilters() {
        addIncludeFilter(new TypeFilter() {
            @Override
            public boolean match(MetadataReader metadataReader, MetadataReaderFactory metadataReaderFactory) throws IOException {
                return true;//所有的类都合格
            }
        });
    }

    // spring的第二次筛选,此筛选后,只剩下了接口类
    @Override
    protected boolean isCandidateComponent(AnnotatedBeanDefinition beanDefinition) {
        //只有接口,返回true
        return beanDefinition.getMetadata().isInterface() && beanDefinition.getMetadata().isIndependent();
    }

    // 扫描mapper
    @Override
    public Set<BeanDefinitionHolder> doScan(String... basePackages) {
        // 通过父类的扫描完成初筛和二筛,此时只剩下了接口类的bd
        Set<BeanDefinitionHolder> beanDefinitions = super.doScan(basePackages);

        // 你可以认为筛选出的这些bd是半成品,不成熟的,还需再加工一下
        processBeanDefinitions(beanDefinitions);
        return beanDefinitions;
    }

    // 加工bd
    private void processBeanDefinitions(Set<BeanDefinitionHolder> beanDefinitions) {
        GenericBeanDefinition definition;
        // 遍历bd, 将bd的class设置成 MapperFactoryBean
        for (BeanDefinitionHolder holder : beanDefinitions) {
            definition = (GenericBeanDefinition) holder.getBeanDefinition();
            
            // 此句的作用是spring在推算构造方法时,会使用带一个参数的构造器,且参数值是mapper接口类
            definition.getConstructorArgumentValues().addGenericArgumentValue(definition.getBeanClassName()); // issue #59
            
            // bd的beanClass是MapperFactoryBean
            definition.setBeanClass(this.mapperFactoryBean.getClass());
            definition.getPropertyValues().add("sqlSessionFactory", this.sqlSessionFactory);
            
            // spring的di时,会扫描bd中beanClass的setter方法,然后注入
            // 此设置最主要的目的是通过set方法,注入我们在配置类(即第1步)中的sqlSessionFactory。
            definition.setAutowireMode(AbstractBeanDefinition.AUTOWIRE_BY_TYPE);
        }
    }
}
  1. 最后了,我们来看MapperFactoryBean
    它是FactoryBean,所以我们直接看它的getObject()方法
public class MapperFactoryBean<T> extends SqlSessionDaoSupport implements FactoryBean<T> {
    private Class<T> mapperInterface;
    
    public MapperFactoryBean(Class<T> mapperInterface) {
        this.mapperInterface = mapperInterface;
    }

    // 哇哦~,它返回了mybatis的mapper,我们继续跟getSqlSession(),这个方法定义在父类SqlSessionDaoSupport中
    @Override
    public T getObject() throws Exception {
        return getSqlSession().getMapper(this.mapperInterface);
    }

    @Override
    public Class<T> getObjectType() {
        return this.mapperInterface;
    }
}
public abstract class SqlSessionDaoSupport extends DaoSupport {
    private SqlSession sqlSession;
    
    // 当注入模型为 AUTOWIRE_BY_TYPE 时,会扫描bd中beanClass的setter方法,
    // 并将我们在配置文件中的sqlSessionFactory注入
    public void setSqlSessionFactory(SqlSessionFactory sqlSessionFactory) {
        if (!this.externalSqlSession) {
            this.sqlSession = new SqlSessionTemplate(sqlSessionFactory);
        }
    }

    // this.sqlSession哪里来的呢,我们发现 setSqlSessionFactory()方法有
    public SqlSession getSqlSession() {
        return this.sqlSession;
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 218,525评论 6 507
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,203评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,862评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,728评论 1 294
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,743评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,590评论 1 305
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,330评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,244评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,693评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,885评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,001评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,723评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,343评论 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,919评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,042评论 1 270
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,191评论 3 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,955评论 2 355

推荐阅读更多精彩内容