Mybatis工作原理簡析

如何解析Mybatis的實現(xiàn)原理

我覺得最簡單的方式看他如何初始化跛锌,這里官方文檔中入門一章已經(jīng)介紹了


SqlSessionFactory實例是Mybatis的核心,而SqlSessionFactory實例又是通過SqlSessionFactoryBuilder獲得柏靶,SqlSesionFactoryBuilder必須從Configuration實例中構(gòu)建出SqlSesionFactory的實例。Configuration實例怎么獲取您炉?從XML配置文件解析完成后各種配置就統(tǒng)一存放在Configuration實例中原在。所以探究Mybatis的工作原理可以順著XML配置文件 —> Configuration實例 —> SqlSesionFactoryBuilder —> SqlSessionFactory實例友扰。
這里有兩種方式:


從 XML 中構(gòu)建 SqlSessionFactory

不使用 XML 構(gòu)建 SqlSessionFactory
  1. 如果按照官方文檔中構(gòu)建SqlSessionFactory實例的兩種方式,直接看SqlSessionFactoryBuilder
  2. 如果使用Mybatis-spring 配置Mybatis庶柿,那就從SqlSessionFactoryBean為入口村怪,這也是最常用的方式,其實里面也是使用官方文檔中的方式浮庐,只是為了便于集成到Spring中甚负,把它包裝成bean的形式

看兩種方式的源碼

//實現(xiàn)了FactoryBean<SqlSessionFactory>,說明該實例返回的實例及類型是SqlSessionFactory,而不是SqlSessionFactoryBean
//實現(xiàn)了InitializingBean梭域,說明屬性值set完后會調(diào)用afterPropertiesSet方法初始化
//實現(xiàn)了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");
    //默認已經(jīng)創(chuàng)建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");
    //構(gòu)建sqlSessionFactory 實例
    this.sqlSessionFactory = buildSqlSessionFactory();
  }
   protected SqlSessionFactory buildSqlSessionFactory() throws IOException {
     if (this.configuration != null) {
       //設(shè)置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);
    }
    //設(shè)置objectFactory
    if (this.objectFactory != null) {
      configuration.setObjectFactory(this.objectFactory);
    }
    //設(shè)置objectWrapperFactory
    if (this.objectWrapperFactory != null) {
      configuration.setObjectWrapperFactory(this.objectWrapperFactory);
    }
    //設(shè)置vfs 
    if (this.vfs != null) {
      configuration.setVfsImpl(this.vfs);
    }
    //設(shè)置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);
      }
    }
    //注冊類型別名赎懦,這里的類型如果與上面的重復(fù)會拋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);
      }
    }
    //注冊類型處理器類,不能與上面有重復(fù)
    if (!isEmpty(this.typeHandlers)) {
      for (TypeHandler<?> typeHandler : this.typeHandlers) {
        configuration.getTypeHandlerRegistry().register(typeHandler);
      }
    }
    //注冊數(shù)據(jù)庫產(chǎn)品名稱Provider循衰,針對不同數(shù)據(jù)庫產(chǎn)品的靈活配置
    if (this.databaseIdProvider != null) {
      try {
        configuration.setDatabaseId(this.databaseIdProvider.getDatabaseId(this.dataSource));
      } catch (SQLException e) {
        throw new NestedIOException("Failed getting a databaseId", e);
      }
    }
    //設(shè)置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();
      }
    }  
    //事務(wù)工廠,默認使用Spring的SpringManagedTransactionFactory
    if (this.transactionFactory == null) {
      this.transactionFactory = new SpringManagedTransactionFactory();
    }
    //設(shè)置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算是創(chuàng)建成功了铲敛,那下面就分析下上面配置中那些注意點和說明點

  1. Variables作用
    它的作用就相當于一個全局常量表,在解析config和mapper文件中会钝,遇到${}這種表達式的都會伐蒋,在常量表中查找,找到就替換迁酸,沒有就保持原樣先鱼。
    它的來源主要在config文件的<properties />節(jié)點中聲明和SqlSessionFactoryBean中configurationProperties屬性值。
  2. objectFactory作用
    用戶創(chuàng)建對象奸鬓,比如傳入List.class焙畔,利用反射返回ArrayList的實例,默認實現(xiàn)類DefaultObjectFactory
  3. objectWrapperFactory作用
    默認實現(xiàn)類DefaultObjectWrapperFactory串远,包裝Object實例宏多,很少使用
  4. vfs作用
    默認實現(xiàn)類DefaultVFS,比如遞歸出給定URL標識的所有資源的全部路徑澡罚,如VFS.getInstance().list(".")伸但,傳入相對路徑。
  5. typeAliasesPackage和typeAliases
    類型別名留搔,前者只需指定包更胖,后者需類全名,指定別名后隔显,在mapper文件和config文件中就可以使用該別名代表類全名了却妨,建議兩者只用一種,如果都用括眠,類名不要重復(fù)彪标。Mybatis默認注冊了很多別名,具體見org.apache.ibatis.session.Configuration
  6. plugins
    插件,用于在執(zhí)行相關(guān)操作時掷豺,在執(zhí)行前后插入自定義行為捞烟,如緩存账锹,分頁等,可指定多個插件坷襟,每個插件可攔截指定方法。
  7. typeHandlersPackage和typeHandlers
    類型處理器生年,Java類型與數(shù)據(jù)庫字段類型之間的轉(zhuǎn)換類婴程,如varchar —> String,Mybatis已經(jīng)注冊了大部分類型處理器抱婉,具體見org.apache.ibatis.type.TypeHandlerRegistry
  8. databaseIdProvider
    MyBatis 可以根據(jù)不同的數(shù)據(jù)庫廠商執(zhí)行不同的語句档叔,這種多廠商的支持是基于映射語句中的 databaseId 屬性,具體見官方文檔說明蒸绩。
  9. cache
    二級緩存衙四,Mybatis默認開啟二級緩存,在一次session中緩存查詢的語句和結(jié)果患亿,默認PerpetualCache實現(xiàn)传蹈,采用LRU方式(LruCache),在mapper文件中可通過<cache />進行設(shè)置
  10. XMLConfigBuilder解析過程中注意點
    <settings /> 元素中的默認值見settingsElement方法
    <properties />元素resource和url不能同時設(shè)置步藕,否則拋異常惦界,該元素中的配置項全局有效
    <typeAliases />package 和typeAlias只能一個有效,package優(yōu)先
    <plugins /> 可以配置多個咙冗,每個屬性值會調(diào)用setProperties方法傳入
    <environments /> 如果配置了默認使用default中定義的environment沾歪,這里建議采用表達式定義,值放在properties文件中雾消,生產(chǎn)灾搏,測試各一份,自動根據(jù)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變量中,實際調(diào)用中mybatis會采用動態(tài)代理方式生成代理實例(詳見MapperProxyFactory)添忘,代理實例和方法調(diào)用過一次就會被緩存起來不會重復(fù)生成采呐,由SqlSession調(diào)用相關(guān)method(詳見MapperProxy和MapperMethod)進行具體數(shù)據(jù)庫操作
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市搁骑,隨后出現(xiàn)的幾起案子斧吐,更是在濱河造成了極大的恐慌又固,老刑警劉巖,帶你破解...
    沈念sama閱讀 212,332評論 6 493
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件煤率,死亡現(xiàn)場離奇詭異仰冠,居然都是意外死亡,警方通過查閱死者的電腦和手機蝶糯,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,508評論 3 385
  • 文/潘曉璐 我一進店門洋只,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人昼捍,你說我怎么就攤上這事识虚。” “怎么了妒茬?”我有些...
    開封第一講書人閱讀 157,812評論 0 348
  • 文/不壞的土叔 我叫張陵担锤,是天一觀的道長。 經(jīng)常有香客問我乍钻,道長肛循,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,607評論 1 284
  • 正文 為了忘掉前任团赁,我火速辦了婚禮育拨,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘欢摄。我一直安慰自己熬丧,他們只是感情好,可當我...
    茶點故事閱讀 65,728評論 6 386
  • 文/花漫 我一把揭開白布怀挠。 她就那樣靜靜地躺著析蝴,像睡著了一般。 火紅的嫁衣襯著肌膚如雪绿淋。 梳的紋絲不亂的頭發(fā)上闷畸,一...
    開封第一講書人閱讀 49,919評論 1 290
  • 那天,我揣著相機與錄音吞滞,去河邊找鬼佑菩。 笑死,一個胖子當著我的面吹牛裁赠,可吹牛的內(nèi)容都是我干的殿漠。 我是一名探鬼主播,決...
    沈念sama閱讀 39,071評論 3 410
  • 文/蒼蘭香墨 我猛地睜開眼佩捞,長吁一口氣:“原來是場噩夢啊……” “哼绞幌!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起一忱,我...
    開封第一講書人閱讀 37,802評論 0 268
  • 序言:老撾萬榮一對情侶失蹤莲蜘,失蹤者是張志新(化名)和其女友劉穎谭确,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體票渠,經(jīng)...
    沈念sama閱讀 44,256評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡逐哈,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,576評論 2 327
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了问顷。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片鞠眉。...
    茶點故事閱讀 38,712評論 1 341
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖择诈,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情出皇,我是刑警寧澤羞芍,帶...
    沈念sama閱讀 34,389評論 4 332
  • 正文 年R本政府宣布,位于F島的核電站郊艘,受9級特大地震影響荷科,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜纱注,卻給世界環(huán)境...
    茶點故事閱讀 40,032評論 3 316
  • 文/蒙蒙 一畏浆、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧狞贱,春花似錦刻获、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,798評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至氧枣,卻和暖如春沐兵,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背便监。 一陣腳步聲響...
    開封第一講書人閱讀 32,026評論 1 266
  • 我被黑心中介騙來泰國打工扎谎, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人烧董。 一個月前我還...
    沈念sama閱讀 46,473評論 2 360
  • 正文 我出身青樓毁靶,卻偏偏與公主長得像,于是被迫代替她去往敵國和親解藻。 傳聞我的和親對象是個殘疾皇子老充,可洞房花燭夜當晚...
    茶點故事閱讀 43,606評論 2 350

推薦閱讀更多精彩內(nèi)容

  • 1. 簡介 1.1 什么是 MyBatis ? MyBatis 是支持定制化 SQL螟左、存儲過程以及高級映射的優(yōu)秀的...
    笨鳥慢飛閱讀 5,461評論 0 4
  • 1 引言# 本文主要講解JDBC怎么演變到Mybatis的漸變過程啡浊,重點講解了為什么要將JDBC封裝成Mybait...
    七寸知架構(gòu)閱讀 76,442評論 36 980
  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理觅够,服務(wù)發(fā)現(xiàn),斷路器巷嚣,智...
    卡卡羅2017閱讀 134,637評論 18 139
  • Java數(shù)據(jù)持久化之mybatis 一. mybatis簡介 1.1 原始的JDBC操作: Java 通過 Jav...
    小Q逛逛閱讀 4,903評論 0 16
  • 單獨使用mybatis是有很多限制的(比如無法實現(xiàn)跨越多個session的事務(wù))喘先,而且很多業(yè)務(wù)系統(tǒng)本來就是使用sp...
    七寸知架構(gòu)閱讀 3,441評論 0 53