Mybatis工作原理简析

如何解析Mybatis的实现原理

我觉得最简单的方式看他如何初始化,这里官方文档中入门一章已经介绍了


SqlSessionFactory实例是Mybatis的核心,而SqlSessionFactory实例又是通过SqlSessionFactoryBuilder获得,SqlSesionFactoryBuilder必须从Configuration实例中构建出SqlSesionFactory的实例。Configuration实例怎么获取?从XML配置文件解析完成后各种配置就统一存放在Configuration实例中。所以探究Mybatis的工作原理可以顺着XML配置文件 —> Configuration实例 —> SqlSesionFactoryBuilder —> SqlSessionFactory实例。
这里有两种方式:


从 XML 中构建 SqlSessionFactory

不使用 XML 构建 SqlSessionFactory
  1. 如果按照官方文档中构建SqlSessionFactory实例的两种方式,直接看SqlSessionFactoryBuilder
  2. 如果使用Mybatis-spring 配置Mybatis,那就从SqlSessionFactoryBean为入口,这也是最常用的方式,其实里面也是使用官方文档中的方式,只是为了便于集成到Spring中,把它包装成bean的形式

看两种方式的源码

//实现了FactoryBean<SqlSessionFactory>,说明该实例返回的实例及类型是SqlSessionFactory,而不是SqlSessionFactoryBean
//实现了InitializingBean,说明属性值set完后会调用afterPropertiesSet方法初始化
//实现了ApplicationListener<ApplicationEvent>,说明会将ApplicationEvent注入到该bean中
public class SqlSessionFactoryBean implements FactoryBean<SqlSessionFactory>, InitializingBean, ApplicationListener<ApplicationEvent> {
    //省略set方法
     @Override
  public void afterPropertiesSet() throws Exception {
    //必须配置dataSource
    notNull(dataSource, "Property 'dataSource' is required");
    //默认已经创建SqlSessionFactoryBuilder实例,一般不用配置
    notNull(sqlSessionFactoryBuilder, "Property 'sqlSessionFactoryBuilder' is required");
    //configuration实例和configLocation不能同时配置
    state((configuration == null && configLocation == null) || !(configuration != null && configLocation != null),
              "Property 'configuration' and 'configLocation' can not specified with together");
    //构建sqlSessionFactory 实例
    this.sqlSessionFactory = buildSqlSessionFactory();
  }
   protected SqlSessionFactory buildSqlSessionFactory() throws IOException {
     if (this.configuration != null) {
       //设置Variable,Variable的作用后面再讲
       configuration = this.configuration;
      if (configuration.getVariables() == null) {
        configuration.setVariables(this.configurationProperties);
      } else if (this.configurationProperties != null) {
        configuration.getVariables().putAll(this.configurationProperties);
      }
    } else if (this.configLocation != null){
      //解析mybatis-config.xml
      xmlConfigBuilder = new XMLConfigBuilder(this.configLocation.getInputStream(), null, this.configurationProperties);
      configuration = xmlConfigBuilder.getConfiguration();
    }else {
      configuration = new Configuration();
      configuration.setVariables(this.configurationProperties);
    }
    //设置objectFactory
    if (this.objectFactory != null) {
      configuration.setObjectFactory(this.objectFactory);
    }
    //设置objectWrapperFactory
    if (this.objectWrapperFactory != null) {
      configuration.setObjectWrapperFactory(this.objectWrapperFactory);
    }
    //设置vfs 
    if (this.vfs != null) {
      configuration.setVfsImpl(this.vfs);
    }
    //设置typeAliasesPackage,类型别名包
    if (hasLength(this.typeAliasesPackage)) {
         //多个包名可以用,或;或制表符或换行符分隔
        String[] typeAliasPackageArray = tokenizeToStringArray(this.typeAliasesPackage,
          ConfigurableApplicationContext.CONFIG_LOCATION_DELIMITERS);
      //注册别名包
      for (String packageToScan : typeAliasPackageArray) {
        configuration.getTypeAliasRegistry().registerAliases(packageToScan,
                typeAliasesSuperType == null ? Object.class : typeAliasesSuperType);
      }
    }
    //注册类型别名,这里的类型如果与上面的重复会抛TypeException异常
    if (!isEmpty(this.typeAliases)) {
      for (Class<?> typeAlias : this.typeAliases) {
        configuration.getTypeAliasRegistry().registerAlias(typeAlias);
      }
    }
    //注册插件
    if (!isEmpty(this.plugins)) {
      for (Interceptor plugin : this.plugins) {
        configuration.addInterceptor(plugin);
      }
    }
    //注册类型处理器包名
    if (hasLength(this.typeHandlersPackage)) {
      String[] typeHandlersPackageArray = tokenizeToStringArray(this.typeHandlersPackage,
          ConfigurableApplicationContext.CONFIG_LOCATION_DELIMITERS);
      for (String packageToScan : typeHandlersPackageArray) {
        configuration.getTypeHandlerRegistry().register(packageToScan);
      }
    }
    //注册类型处理器类,不能与上面有重复
    if (!isEmpty(this.typeHandlers)) {
      for (TypeHandler<?> typeHandler : this.typeHandlers) {
        configuration.getTypeHandlerRegistry().register(typeHandler);
      }
    }
    //注册数据库产品名称Provider,针对不同数据库产品的灵活配置
    if (this.databaseIdProvider != null) {
      try {
        configuration.setDatabaseId(this.databaseIdProvider.getDatabaseId(this.dataSource));
      } catch (SQLException e) {
        throw new NestedIOException("Failed getting a databaseId", e);
      }
    }
    //设置cache
    if (this.cache != null) {
      configuration.addCache(this.cache);
    }
    //解析mybatis-config.xml文件
    if (xmlConfigBuilder != null) {
      try {
        xmlConfigBuilder.parse();
      } catch (Exception ex) {
        throw new NestedIOException("Failed to parse config resource: " + this.configLocation, ex);
      } finally {
        ErrorContext.instance().reset();
      }
    }  
    //事务工厂,默认使用Spring的SpringManagedTransactionFactory
    if (this.transactionFactory == null) {
      this.transactionFactory = new SpringManagedTransactionFactory();
    }
    //设置Environment
    configuration.setEnvironment(new Environment(this.environment, this.transactionFactory, this.dataSource));
    if (!isEmpty(this.mapperLocations)) {
      for (Resource mapperLocation : this.mapperLocations) {
        if (mapperLocation == null) {
          continue;
        } 
        try {
          //解析Mapper文件
          XMLMapperBuilder xmlMapperBuilder = new XMLMapperBuilder(mapperLocation.getInputStream(),
              configuration, mapperLocation.toString(), configuration.getSqlFragments());
          xmlMapperBuilder.parse();
        } catch (Exception e) {
          throw new NestedIOException("Failed to parse mapping resource: '" + mapperLocation + "'", e);
        } finally {
          ErrorContext.instance().reset();
        }
      }
    }
  return this.sqlSessionFactoryBuilder.build(configuration);
}

到这里sqlSessionFactory算是创建成功了,那下面就分析下上面配置中那些注意点和说明点

  1. Variables作用
    它的作用就相当于一个全局常量表,在解析config和mapper文件中,遇到${}这种表达式的都会,在常量表中查找,找到就替换,没有就保持原样。
    它的来源主要在config文件的<properties />节点中声明和SqlSessionFactoryBean中configurationProperties属性值。
  2. objectFactory作用
    用户创建对象,比如传入List.class,利用反射返回ArrayList的实例,默认实现类DefaultObjectFactory
  3. objectWrapperFactory作用
    默认实现类DefaultObjectWrapperFactory,包装Object实例,很少使用
  4. vfs作用
    默认实现类DefaultVFS,比如递归出给定URL标识的所有资源的全部路径,如VFS.getInstance().list("."),传入相对路径。
  5. typeAliasesPackage和typeAliases
    类型别名,前者只需指定包,后者需类全名,指定别名后,在mapper文件和config文件中就可以使用该别名代表类全名了,建议两者只用一种,如果都用,类名不要重复。Mybatis默认注册了很多别名,具体见org.apache.ibatis.session.Configuration
  6. plugins
    插件,用于在执行相关操作时,在执行前后插入自定义行为,如缓存,分页等,可指定多个插件,每个插件可拦截指定方法。
  7. typeHandlersPackage和typeHandlers
    类型处理器,Java类型与数据库字段类型之间的转换类,如varchar —> String,Mybatis已经注册了大部分类型处理器,具体见org.apache.ibatis.type.TypeHandlerRegistry
  8. databaseIdProvider
    MyBatis 可以根据不同的数据库厂商执行不同的语句,这种多厂商的支持是基于映射语句中的 databaseId 属性,具体见官方文档说明。
  9. cache
    二级缓存,Mybatis默认开启二级缓存,在一次session中缓存查询的语句和结果,默认PerpetualCache实现,采用LRU方式(LruCache),在mapper文件中可通过<cache />进行设置
  10. XMLConfigBuilder解析过程中注意点
    <settings /> 元素中的默认值见settingsElement方法
    <properties />元素resource和url不能同时设置,否则抛异常,该元素中的配置项全局有效
    <typeAliases />package 和typeAlias只能一个有效,package优先
    <plugins /> 可以配置多个,每个属性值会调用setProperties方法传入
    <environments /> 如果配置了默认使用default中定义的environment,这里建议采用表达式定义,值放在properties文件中,生产,测试各一份,自动根据properties文件切换。
    <typeHandlers />package和typeHandler同时配置只有一个有效,默认package
    <mappers/> package和mapper同时配置只有一个生效,默认package
  11. mapperLocations与Mapper文件解析
    mapperLocations 中可定义classpath:mapper/*.xml这样的值,spring会找到每个xml文件并进行解析。Mapper文件中select|insert|update|delete以XMLStatementBuilder类的形式保存在configuration的incompleteStatements变量中,cacheRef 保存在configuration的cacheRef中,namespace|cache|parameterMap|resultMap|sql 保存在builderAssistant(XMLMapperBuilder类)中,Mapper接口保存在mapperRegistry变量中,实际调用中mybatis会采用动态代理方式生成代理实例(详见MapperProxyFactory),代理实例和方法调用过一次就会被缓存起来不会重复生成,由SqlSession调用相关method(详见MapperProxy和MapperMethod)进行具体数据库操作
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,132评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,802评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,566评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,858评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,867评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,695评论 1 282
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,064评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,705评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,915评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,677评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,796评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,432评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,041评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,992评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,223评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,185评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,535评论 2 343

推荐阅读更多精彩内容

  • 1. 简介 1.1 什么是 MyBatis ? MyBatis 是支持定制化 SQL、存储过程以及高级映射的优秀的...
    笨鸟慢飞阅读 5,423评论 0 4
  • 1 引言# 本文主要讲解JDBC怎么演变到Mybatis的渐变过程,重点讲解了为什么要将JDBC封装成Mybait...
    七寸知架构阅读 76,423评论 36 980
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,598评论 18 139
  • Java数据持久化之mybatis 一. mybatis简介 1.1 原始的JDBC操作: Java 通过 Jav...
    小Q逛逛阅读 4,897评论 0 16
  • 单独使用mybatis是有很多限制的(比如无法实现跨越多个session的事务),而且很多业务系统本来就是使用sp...
    七寸知架构阅读 3,428评论 0 53