欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Spring Mybatis 分頁(yè)插件使用教程

 更新時(shí)間:2018年02月23日 17:06:18   作者:南柯問(wèn)天  
這篇文章主要介紹了Spring Mybatis分頁(yè)插件使用教程,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下

Mybatis分頁(yè)切入點(diǎn)

Mybatis內(nèi)部有個(gè)plugins(插件)概念,本質(zhì)上屬于攔截器的思想。具體的解析可見(jiàn)他文MyBatis攔截器原理探究。本文將在此基礎(chǔ)上直接展示實(shí)際項(xiàng)目的實(shí)現(xiàn)代碼和其他的相關(guān)解析

分頁(yè)具體代碼實(shí)現(xiàn)

首先我們可以定義方言抽象類,用于實(shí)現(xiàn)分頁(yè)AbstractDialect.java

public abstract class AbstractDialect{
 /**
  * 是否支持limit和偏移量
  * @return
  */
 public abstract boolean supportsLimitOffset();
 /**
  * 是否支持limit
  * @return
  */
 public abstract boolean supportsLimit();
 /**
  * 獲取增加了分頁(yè)屬性之后的SQL
  * @param sql
  * @param offset
  * @param limit
  * @return
  */
 public abstract String getLimitString(String sql, int offset, int limit);
}

再而我們就以O(shè)racle與Mysql數(shù)據(jù)庫(kù)的分頁(yè)技術(shù)作下分別的實(shí)現(xiàn)

MySQLDialect.java-Mysql分頁(yè)方言

public class MySQLDialect extends AbstractDialect {
 public boolean supportsLimitOffset() {
  return true;
 }
 public boolean supportsLimit() {
  return true;
 }
 public String getLimitString(String sql, int offset, int limit) {
  if (offset > 0) {
   return sql + " limit " + offset + "," + limit;
  } else {
   return sql + " limit " + limit;
  }
 }
}

OracleDialect.java-Oracle方言實(shí)現(xiàn)

public class OracleDialect extends ADialect{
 @Override
 public boolean supportsLimitOffset() {
  return false;
 }
 @Override
 public boolean supportsLimit() {
  return false;
 }
 @Override
 public String getLimitString(String sql, int start, int limit) {
  if(start < 0){
   start = 0;
  }
  if(limit < 0){
   limit = 10;
  }
  StringBuilder pageSql = new StringBuilder(100);
  pageSql.append("select * from ( select temp.*, rownum row_id from ( ");
  pageSql.append(sql);
  pageSql.append(" ) temp where rownum <= ").append(start+limit);
  pageSql.append(") where row_id > ").append(start);
  return pageSql.toString();
 }
}

對(duì)應(yīng)的Mybatis插件攔截器實(shí)現(xiàn)如下,攔截StatementHandler#prepare(Connection con)創(chuàng)建SQL語(yǔ)句對(duì)象方法

PaginationInterceptor.java

@Intercepts({ @Signature(type = StatementHandler.class, method = "prepare", args = { Connection.class }) })
public final class PaginationInterceptor implements Interceptor {
 private final static Logger log = LoggerFactory
  .getLogger(PaginationInterceptor.class);
 private ADialect dialect;
 public void setDialect(ADialect dialect) {
  this.dialect = dialect;
 }
 @Override
 public Object intercept(Invocation invocation) throws Throwable {
  // 直接獲取攔截的對(duì)象,其實(shí)現(xiàn)類RoutingStatementHandler
  StatementHandler statementHandler = (StatementHandler) invocation
   .getTarget();
  BoundSql boundSql = statementHandler.getBoundSql();
  // 獲取元對(duì)象,主要用于獲取statementHandler所關(guān)聯(lián)的對(duì)象及屬性
  MetaObject metaStatementHandler = MetaObject.forObject(
   statementHandler, new DefaultObjectFactory(),
   new DefaultObjectWrapperFactory());
  MappedStatement mappedStmt= (MappedStatement) metaStatementHandler
   .getValue("delegate.mappedStatement".intern());
  // 只對(duì)queryPagination()方法進(jìn)行分頁(yè)操作
  if(mappedStmt.getId().indexOf("queryPagination")==-1){
   return invocation.proceed();
  }
  // 重新構(gòu)造分頁(yè)的sql
  String originalSql = (String) metaStatementHandler
   .getValue("delegate.boundSql.sql".intern());
  metaStatementHandler.setValue("delegate.boundSql.sql".intern(), dialect
   .getLimitString(originalSql, rowBounds.getOffset(),
    rowBounds.getLimit()));
  metaStatementHandler.setValue("delegate.rowBounds.offset".intern(),
   RowBounds.NO_ROW_OFFSET);
  metaStatementHandler.setValue("delegate.rowBounds.limit".intern(),
   RowBounds.NO_ROW_LIMIT);
  log.debug("page sql : " + boundSql.getSql());
  return invocation.proceed();
 }
 // 攔截對(duì)象
 @Override
 public Object plugin(Object target) {
  return Plugin.wrap(target, this);
 }
 @Override
 public void setProperties(Properties properties) {
 }
}

Spring對(duì)應(yīng)的xml配置可如下,以oracle分頁(yè)為例子

<!-- oracle方言配置,用于oracle的分頁(yè) -->
 <bean id="paginationInterceptor"  class="com.hsnet.winner.cas.admin.core.dao.mybatis.interceptor.PaginationInterceptor">
  <property name="dialect">
   <bean class="cn.cloud.winner.oss.manager.mybatis.page.OracleDialect" />
  </property>
 </bean>

使用以上的代碼以及配置即可完成對(duì)oracle數(shù)據(jù)庫(kù)以及mysql數(shù)據(jù)庫(kù)的分頁(yè)操作。并且博主對(duì)其中的某個(gè)點(diǎn)作下解析

Mybatis#MetaObject-元數(shù)據(jù)對(duì)象解析

以上的代碼博主之前在使用的時(shí)候,對(duì)其中的MetaObject這個(gè)類很費(fèi)解,其直接通過(guò)getValue()方法便可以將所代理的對(duì)象的所關(guān)聯(lián)的屬性全都拿取到。我們可以跟隨源碼深入了解下

MetaObject#forObject()

代理對(duì)象均通過(guò)此靜態(tài)方法進(jìn)入

public static MetaObject forObject(Object object, ObjectFactory objectFactory, ObjectWrapperFactory objectWrapperFactory) {
 if (object == null) {
  return SystemMetaObject.NULL_META_OBJECT;
 } else {
  return new MetaObject(object, objectFactory, objectWrapperFactory);
 }
 }

我們可以直接觀察其中的構(gòu)造函數(shù),玄機(jī)就在此處

private MetaObject(Object object, ObjectFactory objectFactory, ObjectWrapperFactory objectWrapperFactory) {
 this.originalObject = object;
 this.objectFactory = objectFactory;
 this.objectWrapperFactory = objectWrapperFactory;
 // 所有的屬性獲取均通過(guò)objectWrapper類來(lái)獲取,此處主要對(duì)所代理的object對(duì)象類型進(jìn)行判斷
 if (object instanceof ObjectWrapper) {
  this.objectWrapper = (ObjectWrapper) object;
 } else if (objectWrapperFactory.hasWrapperFor(object)) {
  this.objectWrapper = objectWrapperFactory.getWrapperFor(this, object);
 } else if (object instanceof Map) {
  this.objectWrapper = new MapWrapper(this, (Map) object);
 } else if (object instanceof Collection) {
  this.objectWrapper = new CollectionWrapper(this, (Collection) object);
 } else {
  // 我們常用的便是BeanWrapper
  this.objectWrapper = new BeanWrapper(this, object);
 }
 }

為了理解的更為滲透,我們繼續(xù)跟進(jìn),最后我們得知其會(huì)調(diào)用Reflector類的構(gòu)造函數(shù)

private Reflector(Class<?> clazz) {
 type = clazz;
 // 獲取構(gòu)造類
 addDefaultConstructor(clazz);
 // 獲取get方法集合
 addGetMethods(clazz);
 // 獲取set方法集合
 addSetMethods(clazz);
 // 獲取內(nèi)部屬性集合
 addFields(clazz);
 readablePropertyNames = getMethods.keySet().toArray(new String[getMethods.keySet().size()]);
 writeablePropertyNames = setMethods.keySet().toArray(new String[setMethods.keySet().size()]);
 for (String propName : readablePropertyNames) {
  caseInsensitivePropertyMap.put(propName.toUpperCase(Locale.ENGLISH), propName);
 }
 for (String propName : writeablePropertyNames) {
  caseInsensitivePropertyMap.put(propName.toUpperCase(Locale.ENGLISH), propName);
 }
 }

由此我們便可知使用Reflector代理類以及MetaObject便可以遍歷代理被代理類的所關(guān)聯(lián)的所有屬性,就拿RoutingStatementHandler類來(lái)說(shuō),經(jīng)過(guò)上述操作后其便可以訪問(wèn)內(nèi)部屬性delegate以及delegate的內(nèi)部屬性configuration/objectFactory/typeHandlerRegistry/resultSetHandler/parameterHandler/mappedStatement等屬性

MetaObject#getValue()

上述闡述的是如何代理被代理類的內(nèi)部屬性,我們也簡(jiǎn)單的看下是如何正確的調(diào)用

public Object getValue(String name) {
 // PropertyTokenizer與StringTokenizer類似,只是前者寫(xiě)死以.為分隔符
 PropertyTokenizer prop = new PropertyTokenizer(name);
 if (prop.hasNext()) {
  MetaObject metaValue = metaObjectForProperty(prop.getIndexedName());
  if (metaValue == SystemMetaObject.NULL_META_OBJECT) {
  return null;
  } else {
  return metaValue.getValue(prop.getChildren());
  }
 } else {
  return objectWrapper.get(prop);
 }
 }

具體的解析就不在此闡述了,如何用戶想獲取StatementHandler所擁有的sql字符串,可通過(guò)getValue("delegate.boundSql.sql")其中以.為分隔符并其中的屬性必須是內(nèi)部屬性(區(qū)分大小寫(xiě))。

MetaObject#setValue()

原理同MetaObject#getValue()方法

總結(jié)

以上所述是小編給大家介紹的Spring Mybatis 分頁(yè)插件使用教程,希望對(duì)大家有所幫助,如果大家有任何疑問(wèn)請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!

相關(guān)文章

  • java將圖片分割為幾個(gè)部分示例

    java將圖片分割為幾個(gè)部分示例

    這篇文章主要介紹了java將圖片分割為幾個(gè)部分示例,需要的朋友可以參考下
    2014-04-04
  • 源碼分析Spring?中?@Qualifier?注解基本用法

    源碼分析Spring?中?@Qualifier?注解基本用法

    這篇文章主要介紹了源碼分析Spring?中?@Qualifier?注解基本用法,在源碼分析的過(guò)程中,也?GET?到?Spring?許多新的玩法,感興趣的小伙伴趕緊去試試吧
    2023-08-08
  • 解決swagger2.9.2接口文檔顯示的問(wèn)題

    解決swagger2.9.2接口文檔顯示的問(wèn)題

    這篇文章主要介紹了解決swagger2.9.2接口文檔顯示的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-06-06
  • mybatis整合ehcache做三級(jí)緩存的實(shí)現(xiàn)方法

    mybatis整合ehcache做三級(jí)緩存的實(shí)現(xiàn)方法

    ehcache是一個(gè)快速內(nèi)存緩存框架,java項(xiàng)目里用起來(lái)很方便,下面這篇文章主要給大家介紹了關(guān)于mybatis整合ehcache做三級(jí)緩存的實(shí)現(xiàn)方法,文中通過(guò)實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2023-06-06
  • SpringBoot+TestNG單元測(cè)試的實(shí)現(xiàn)

    SpringBoot+TestNG單元測(cè)試的實(shí)現(xiàn)

    本文主要介紹了SpringBoot+TestNG單元測(cè)試的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2021-07-07
  • SpringCloud組件之Eureka Server詳細(xì)啟動(dòng)過(guò)程及說(shuō)明

    SpringCloud組件之Eureka Server詳細(xì)啟動(dòng)過(guò)程及說(shuō)明

    這篇文章主要介紹了SpringCloud組件之Eureka Server詳細(xì)啟動(dòng)過(guò)程及說(shuō)明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2024-01-01
  • Java基礎(chǔ)之Maven詳解

    Java基礎(chǔ)之Maven詳解

    這篇文章主要介紹了Java基礎(chǔ)之Maven詳解,文中有非常詳細(xì)的代碼示例,對(duì)正在學(xué)習(xí)java基礎(chǔ)的小伙伴們有非常好的幫助,需要的朋友可以參考下
    2021-04-04
  • Java實(shí)現(xiàn)的打印螺旋矩陣算法示例

    Java實(shí)現(xiàn)的打印螺旋矩陣算法示例

    這篇文章主要介紹了Java實(shí)現(xiàn)的打印螺旋矩陣算法,結(jié)合完整實(shí)例形式詳細(xì)分析了java打印螺旋矩陣的算法原理與實(shí)現(xiàn)技巧,需要的朋友可以參考下
    2019-10-10
  • OpenCV Java實(shí)現(xiàn)人臉識(shí)別和裁剪功能

    OpenCV Java實(shí)現(xiàn)人臉識(shí)別和裁剪功能

    這篇文章主要為大家詳細(xì)介紹了OpenCV Java實(shí)現(xiàn)人臉識(shí)別和裁剪功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2019-05-05
  • java中instanceof和getClass()的區(qū)別分析

    java中instanceof和getClass()的區(qū)別分析

    本篇文章介紹了,在java中instanceof和getClass()的區(qū)別分析。需要的朋友參考下
    2013-04-04

最新評(píng)論