亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 數據庫 > 文庫 > 正文

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

2020-10-29 21:48:24
字體:
來源:轉載
供稿:網友

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

1.1     啟用延遲加載

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

       第一種是在對應的<collection>或<association>標簽上指定fetchType屬性值為“lazy”。如下示例中我們在查詢id為selectByPrimaryKey的查詢時會返回BaseResultMap,在BaseResultMap中,我們指定了屬性“nodes”是一個集合類型的,而且是需要通過id為selectNodes的查詢進行查詢的,我們指定了該查詢的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>標簽下加上如下配置可開啟全局的延遲加載。開啟了全局的延遲加載后我們就無需再在各個嵌套的子查詢上配置延遲加載了,如果有某一個嵌套的子查詢是不需要延遲加載的,可以設置其fetchType=”eager”。設置在嵌套查詢上的fetchType可以覆蓋全局的延遲加載設置。

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

1.2     分析

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

 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先是根據正常情況創建一個返回類型對應的對象。當我們的ResultMap是包含子查詢的時候,其會在我們正常返回類型對象的基礎上創建對應的代理對象。對,你沒有看錯,就是我們的直接結果是代理對象,而不是子查詢對應的屬性是代理對象。默認是基于JavassistProxyFactory類創建的代理對象??梢酝ㄟ^Mybatis的全局配置proxyFactory來更改,可選值是CGLIB和JAVASSIST,默認是后者。需要使用CGLIB代理時注意加入CGLIB的包。

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

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

 @Test  public void testLazyLoad1() {   SysWfProcessMapper mapper = this.session.getMapper(SysWfProcessMapper.class);   SysWfProcess process = mapper.selectByPrimaryKey(1);   System.out.println(process.getClass());  } 

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

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: 1class com.elim.learn.mybatis.model.SysWfProcess_$$_jvstc25_02016-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()去掉,也就是說我們只是從數據庫中查詢出SysWfProcess對象,而不使用它的時候,通過查看日志輸出你會發現Mybatis又只會發送一條SQL,即只是查詢出SysWfProcess的信息。這是為什么呢?

 1.3     aggressiveLazyLoading

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

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

1.4     lazyLoadTriggerMethods

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

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

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

@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);   }  } }

       本文是介紹的都是基于<collection>這種關聯,其實<association>關聯的對象的延遲加載也是一樣的,它們的默認策略也是一樣的。

感謝閱讀,希望能幫助到大家,謝謝大家對本站的支持!

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
91精品国产乱码久久久久久蜜臀| 国产精品直播网红| 国产一区二区欧美日韩| 色综合久久精品亚洲国产| 黄色成人在线免费| 欧美国产亚洲精品久久久8v| 久久中文精品视频| 亚洲男人天堂久| 色老头一区二区三区| 久久久久久有精品国产| 亚洲国产精品yw在线观看| 久热爱精品视频线路一| 综合av色偷偷网| 精品中文字幕久久久久久| 国产成人啪精品视频免费网| 亚洲一区亚洲二区| 亚洲一区第一页| 日韩精品欧美国产精品忘忧草| 欧美亚洲日本网站| 中文字幕欧美亚洲| 欧美成人在线免费视频| 日本韩国欧美精品大片卡二| 欧美激情欧美激情在线五月| 久久久久久久久久国产| 亚洲最大av网| 国产亚洲视频中文字幕视频| 亚洲精品综合久久中文字幕| 欧美亚洲第一区| 最近更新的2019中文字幕| 国产精品久久久久久久久借妻| 福利视频导航一区| 亚洲精品久久久久久下一站| 国产成人精品电影| 国产一区二区三区视频| 亚洲精品在线看| 成人伊人精品色xxxx视频| 国产成人精品a视频一区www| 国产精品一区二区三区久久| 日韩在线小视频| 成人免费视频网址| 国产国产精品人在线视| 国产激情久久久久| 91成人天堂久久成人| 中文字幕在线国产精品| 国产精品久久视频| 久久精品国产免费观看| 亚洲欧美一区二区三区在线| 欧美激情奇米色| 久久久综合免费视频| 4388成人网| 久久综合伊人77777蜜臀| 成人免费观看a| 最近的2019中文字幕免费一页| 国产97在线播放| 伦理中文字幕亚洲| 欧美日韩在线视频一区| 日韩中文字幕亚洲| 一区二区三区美女xx视频| 国产丝袜一区二区| 日本19禁啪啪免费观看www| 91精品国产91久久久| 亚洲国产精品人久久电影| 亚洲国产精品推荐| 韩国视频理论视频久久| 欧美精品九九久久| 精品久久久999| 在线日韩中文字幕| 色七七影院综合| 亚洲天堂色网站| 亚洲毛茸茸少妇高潮呻吟| 日韩中文视频免费在线观看| 在线观看日韩专区| 欧美有码在线观看视频| 国产欧美精品一区二区| 欧美亚洲一区在线| 欧美另类高清videos| 亚洲桃花岛网站| 中文字幕亚洲无线码a| 欧美成人黑人xx视频免费观看| 午夜剧场成人观在线视频免费观看| 国产精品视频yy9099| 成人性生交大片免费看小说| 欧美精品久久久久久久久| 日韩欧美在线播放| 国产欧美久久久久久| 日韩精品视频在线观看网址| 国产精品亚洲片夜色在线| 亚洲wwwav| 国产亚洲精品一区二区| 另类少妇人与禽zozz0性伦| 这里只有精品丝袜| 日韩中文视频免费在线观看| 九九久久综合网站| 亚洲人成电影在线播放| 中文字幕欧美亚洲| 欧美高清videos高潮hd| 亚洲成av人乱码色午夜| 日韩在线观看免费高清| 人体精品一二三区| 日韩精品有码在线观看| 亚洲精品xxx| 中文字幕国产亚洲| 日韩视频免费在线| 精品久久久久久久久久国产| 成人a视频在线观看| 国产亚洲a∨片在线观看| 成人看片人aa| 精品国产1区2区| 亚洲高清av在线| 久久精品成人欧美大片古装| 国产一区二区精品丝袜| 在线观看中文字幕亚洲| 日本精品视频在线| 欧美老女人在线视频| 深夜福利日韩在线看| 欧美视频在线视频| 亚洲成色777777女色窝| 国产在线a不卡| 57pao国产精品一区| 欧美激情在线视频二区| 日韩成人在线视频网站| 亚洲国内高清视频| 57pao国产精品一区| 欧美裸身视频免费观看| 日韩美女免费观看| 亚洲精品国产电影| 97av在线视频| 色综合天天狠天天透天天伊人| 国产精品免费视频久久久| 色诱女教师一区二区三区| 国产精品自拍偷拍| 91久久精品国产91性色| 欧美亚洲国产日本| 日韩精品www| 国产亚洲激情视频在线| 欧美老少配视频| 日本19禁啪啪免费观看www| 亚洲国产精品热久久| 久久久免费高清电视剧观看| 久久久精品一区二区三区| 精品国产一区二区三区久久狼5月| 亚洲第一页中文字幕| 亚洲91精品在线观看| 久久久亚洲网站| 成人黄色短视频在线观看| 亚洲一二三在线| 国产精品久久久久久久久免费| 亚洲欧美三级在线| 91高清视频免费观看| 亚洲大胆人体视频| 精品国产电影一区| 中文字幕精品视频| 丝袜亚洲另类欧美重口| 亚洲在线视频观看| 国产成人91久久精品| 精品无人国产偷自产在线| 亚洲精品国产精品国自产观看浪潮| 欧美一级高清免费| 不卡在线观看电视剧完整版| 日韩av影院在线观看| 不卡av日日日| 久久久久久亚洲精品中文字幕| 51精品国产黑色丝袜高跟鞋| 欧美一级淫片丝袜脚交|