Mybatis查詢延遲加載的示例分析-創(chuàng)新互聯(lián)

小編給大家分享一下Mybatis查詢延遲加載的示例分析,相信大部分人都還不怎么了解,因此分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后大有收獲,下面讓我們一起去了解一下吧!

發(fā)展壯大離不開廣大客戶長期以來的信賴與支持,我們將始終秉承“誠信為本、服務(wù)至上”的服務(wù)理念,堅持“二合一”的優(yōu)良服務(wù)模式,真誠服務(wù)每家企業(yè),認(rèn)真做好每個細(xì)節(jié),不斷完善自我,成就企業(yè),實現(xiàn)共贏。行業(yè)涉及成都三輪攪拌車等,在重慶網(wǎng)站建設(shè)公司、成都全網(wǎng)營銷推廣、WAP手機(jī)網(wǎng)站、VI設(shè)計、軟件開發(fā)等項目上具有豐富的設(shè)計經(jīng)驗。

Mybatis查詢延遲加載詳解及實例

1.1     啟用延遲加載

       Mybatis的延遲加載是針對嵌套查詢而言的,是指在進(jìn)行查詢的時候先只查詢最外層的SQL,對于內(nèi)層SQL將在需要使用的時候才查詢出來。Mybatis的延遲加載默認(rèn)是關(guān)閉的,即默認(rèn)是一次就將所有的嵌套SQL一并查了將對象所有的信息都查詢出來。開啟延遲加載有兩種方式。

       第一種是在對應(yīng)的<collection>或<association>標(biāo)簽上指定fetchType屬性值為“l(fā)azy”。如下示例中我們在查詢id為selectByPrimaryKey的查詢時會返回BaseResultMap,在BaseResultMap中,我們指定了屬性“nodes”是一個集合類型的,而且是需要通過id為selectNodes的查詢進(jìn)行查詢的,我們指定了該查詢的fetchType為lazy,即延遲加載。

  <resultMap id="BaseResultMap" type="com.elim.learn.mybatis.model.SysWfProcess">

   <id column="id" jdbcType="INTEGER" property="id" />

   <result column="template_id" jdbcType="INTEGER" property="templateId" />

   <result column="creator" jdbcType="INTEGER" property="creator" />

   <result column="create_time" jdbcType="TIMESTAMP" property="createTime" />

   <collection property="nodes" column="id"

    ofType="com.elim.learn.mybatis.model.SysWfNode" select="selectNodes" fetchType="lazy"/>

  </resultMap>

  <resultMap id="SysWfNodeResult" type="com.elim.learn.mybatis.model.SysWfNode">

   <id column="id" jdbcType="INTEGER" property="nodeId" />

   <result column="process_id" jdbcType="INTEGER" property="processId" />

   <result column="node_code" jdbcType="VARCHAR" property="nodeCode" />

   <result column="node_name" jdbcType="VARCHAR" property="nodeName" />

  </resultMap>

  <select id="selectByPrimaryKey" parameterType="java.lang.Integer"

   resultMap="BaseResultMap">

   select

   <include refid="Base_Column_List" />

   from sys_wf_process

   where id = #{id,jdbcType=INTEGER}

  </select>

  <select id="selectNodes"

   resultMap="SysWfNodeResult">

   select id, process_id, node_code, node_name from sys_wf_node

   where process_id=#{id}

  </select>

       第二種是開啟全局的延遲加載。通過在Mybatis的配置文件的<settings>標(biāo)簽下加上如下配置可開啟全局的延遲加載。開啟了全局的延遲加載后我們就無需再在各個嵌套的子查詢上配置延遲加載了,如果有某一個嵌套的子查詢是不需要延遲加載的,可以設(shè)置其fetchType=”eager”。設(shè)置在嵌套查詢上的fetchType可以覆蓋全局的延遲加載設(shè)置。

   <setting name="lazyLoadingEnabled" value="true"/>

1.2     分析

       Mybatis的查詢結(jié)果是由ResultSetHandler接口的handleResultSets()方法處理的。ResultSetHandler接口只有一個實現(xiàn),DefaultResultSetHandler。有興趣的朋友可以去看一下它的源碼,看一下它是如何處理結(jié)果集的。對于本文的主題,延遲加載相關(guān)的一個核心的方法就是如下這個創(chuàng)建返回結(jié)果對象的方法。

 private Object createResultObject(ResultSetWrapper rsw, ResultMap resultMap, ResultLoaderMap lazyLoader, String columnPrefix) throws SQLException {

  final List<Class<?>> constructorArgTypes = new ArrayList<Class<?>>();

  final List<Object> constructorArgs = new ArrayList<Object>();

  final Object resultObject = createResultObject(rsw, resultMap, constructorArgTypes, constructorArgs, columnPrefix);

  if (resultObject != null && !typeHandlerRegistry.hasTypeHandler(resultMap.getType())) {

   final List<ResultMapping> propertyMappings = resultMap.getPropertyResultMappings();

   for (ResultMapping propertyMapping : propertyMappings) {

    // issue gcode #109 && issue #149

    if (propertyMapping.getNestedQueryId() != null && propertyMapping.isLazy()) {

     return configuration.getProxyFactory().createProxy(resultObject, lazyLoader, configuration, objectFactory, constructorArgTypes, constructorArgs);

    }

   }

  }

  return resultObject;

 }

        在上面方法中我們可以看到Mybatis先是根據(jù)正常情況創(chuàng)建一個返回類型對應(yīng)的對象。當(dāng)我們的ResultMap是包含子查詢的時候,其會在我們正常返回類型對象的基礎(chǔ)上創(chuàng)建對應(yīng)的代理對象。對,你沒有看錯,就是我們的直接結(jié)果是代理對象,而不是子查詢對應(yīng)的屬性是代理對象。默認(rèn)是基于JavassistProxyFactory類創(chuàng)建的代理對象??梢酝ㄟ^Mybatis的全局配置proxyFactory來更改,可選值是CGLIB和JAVASSIST,默認(rèn)是后者。需要使用CGLIB代理時注意加入CGLIB的包。

   <setting name="proxyFactory" value="CGLIB"/>

       回過頭來看我們之前的那個延遲加載的配置,我們的一個查詢返回的是SysWfProcess類型的對象,其有一個SysWfNode集合類型的nodes屬性,nodes屬性是通過一個子查詢查出來的,而且是延遲加載。這個時候我們來進(jìn)行以下測試。

 @Test

  public void testLazyLoad1() {

   SysWfProcessMapper mapper = this.session.getMapper(SysWfProcessMapper.class);

   SysWfProcess process = mapper.selectByPrimaryKey(1);

   System.out.println(process.getClass());

  }

       這個時候你會發(fā)現(xiàn),上面的測試代碼的輸出結(jié)果是一個代理類,而不是我們自己的com.elim.learn.mybatis.model.SysWfProcess類型。另外如果你啟用了日志輸出,并且是打印的DEBUG日志,你會看到Mybatis是發(fā)了兩條SQL進(jìn)行查詢的。

2016-12-23 15:43:21,131 DEBUG [main] (BaseJdbcLogger.java:145) - ==> Preparing: select id, template_id, creator, create_time from sys_wf_process where id = ?

2016-12-23 15:43:21,156 DEBUG [main] (BaseJdbcLogger.java:145) - ==> Parameters: 1(Integer)

2016-12-23 15:43:21,269 DEBUG [main] (BaseJdbcLogger.java:145) - <==   Total: 1

class com.elim.learn.mybatis.model.SysWfProcess_$$_jvstc25_0

2016-12-23 15:43:21,271 DEBUG [main] (BaseJdbcLogger.java:145) - ==> Preparing: select id, process_id, node_code, node_name from sys_wf_node where process_id=?

2016-12-23 15:43:21,272 DEBUG [main] (BaseJdbcLogger.java:145) - ==> Parameters: 1(Integer)

2016-12-23 15:43:21,274 DEBUG [main] (BaseJdbcLogger.java:145) - <==   Total: 2

       但是如果我們把最后一個System.out.println()去掉,也就是說我們只是從數(shù)據(jù)庫中查詢出SysWfProcess對象,而不使用它的時候,通過查看日志輸出你會發(fā)現(xiàn)Mybatis又只會發(fā)送一條SQL,即只是查詢出SysWfProcess的信息。這是為什么呢?

 1.3     aggressiveLazyLoading

       這是因為當(dāng)我們啟用了延遲加載時,我們的查詢結(jié)果返回的是一個代理對象,當(dāng)我們訪問該代理對象的方法時,都會觸發(fā)加載所有的延遲加載的對象信息。這也就可以很好的解釋上面的場景。但是如果是這樣的設(shè)計,貌似Mybatis的延遲加載作用不大。但事實并非如此,這只是Mybatis的一個默認(rèn)策略,我們可以通過Mybatis的全局配置aggressiveLazyLoading來改變它,默認(rèn)是true,表示延遲加載時將在第一次訪問代理對象的方法時就將全部的延遲加載對象加載出來。當(dāng)設(shè)置為false時則會在我們第一次訪問延遲加載的對象的時候才會從數(shù)據(jù)庫加載對應(yīng)的數(shù)據(jù)。注意在延遲對象未從數(shù)據(jù)庫加載出來前,我們對應(yīng)延遲對象的屬性將是null,因為你沒有對它賦值。

  <setting name="aggressiveLazyLoading" value="fasle"/>

1.4     lazyLoadTriggerMethods

       那如果我們設(shè)置了aggressiveLazyLoading=”false”,但又希望在調(diào)用某些方法之前把所有的延遲對象都從數(shù)據(jù)庫加載出來,怎么辦呢?這個時候我們可以通過lazyLoadTriggerMethods參數(shù)來指定需要加載延遲對象的方法調(diào)用。默認(rèn)是equals、clone、hashCode和toString,也就是說我們在調(diào)用代理對象的這些方法之前就會把延遲加載對象從數(shù)據(jù)庫加載出來。

   <setting name="lazyLoadTriggerMethods" value="equals,clone,hashCode,toString" />

       Mybatis延遲加載生成的代理對象的代理過程,可以參考ProxyFactory的創(chuàng)建代理對象的過程,以下是基于Javassist創(chuàng)建的代理對象的代理過程,基于CGLIB的代理也是類似的。從下面的代碼我們可以看到Mybatis的代理對象需要從數(shù)據(jù)庫加載延遲對象時是在目標(biāo)方法被調(diào)用以前發(fā)生的,這就可以保證我們的目標(biāo)方法被調(diào)用時延遲加載的對象已經(jīng)從數(shù)據(jù)庫中加載出來了。

@Override

  public Object invoke(Object enhanced, Method method, Method methodProxy, Object[] args) throws Throwable {

   final String methodName = method.getName();

   try {

    synchronized (lazyLoader) {

     if (WRITE_REPLACE_METHOD.equals(methodName)) {

      Object original = null;

      if (constructorArgTypes.isEmpty()) {

       original = objectFactory.create(type);

      } else {

       original = objectFactory.create(type, constructorArgTypes, constructorArgs);

      }

      PropertyCopier.copyBeanProperties(type, enhanced, original);

      if (lazyLoader.size() > 0) {

       return new JavassistSerialStateHolder(original, lazyLoader.getProperties(), objectFactory, constructorArgTypes, constructorArgs);

      } else {

       return original;

      }

     } else {

      if (lazyLoader.size() > 0 && !FINALIZE_METHOD.equals(methodName)) {

       if (aggressive || lazyLoadTriggerMethods.contains(methodName)) {

        lazyLoader.loadAll();

       } else if (PropertyNamer.isProperty(methodName)) {

        final String property = PropertyNamer.methodToProperty(methodName);

        if (lazyLoader.hasLoader(property)) {

         lazyLoader.load(property);

        }

       }

      }

     }

    }

    return methodProxy.invoke(enhanced, args);

   } catch (Throwable t) {

    throw ExceptionUtil.unwrapThrowable(t);

   }

  }

 }

以上是“Mybatis查詢延遲加載的示例分析”這篇文章的所有內(nèi)容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內(nèi)容對大家有所幫助,如果還想學(xué)習(xí)更多知識,歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道!

網(wǎng)站題目:Mybatis查詢延遲加載的示例分析-創(chuàng)新互聯(lián)
當(dāng)前網(wǎng)址:http://muchs.cn/article32/eiosc.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供虛擬主機(jī)、網(wǎng)站內(nèi)鏈App開發(fā)、手機(jī)網(wǎng)站建設(shè)小程序開發(fā)、商城網(wǎng)站

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)

成都網(wǎng)頁設(shè)計公司