激情久久久_欧美视频区_成人av免费_不卡视频一二三区_欧美精品在欧美一区二区少妇_欧美一区二区三区的

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術|正則表達式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務器之家 - 編程語言 - Java教程 - Spring Mybatis 分頁插件使用教程

Spring Mybatis 分頁插件使用教程

2021-04-06 11:55南柯問天 Java教程

這篇文章主要介紹了Spring Mybatis分頁插件使用教程,非常不錯,具有參考借鑒價值,需要的朋友可以參考下

Mybatis分頁切入點

Mybatis內部有個plugins(插件)概念,本質上屬于攔截器的思想。具體的解析可見他文MyBatis攔截器原理探究。本文將在此基礎上直接展示實際項目的實現代碼和其他的相關解析

分頁具體代碼實現

首先我們可以定義方言抽象類,用于實現分頁AbstractDialect.java

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
public abstract class AbstractDialect{
 /**
  * 是否支持limit和偏移量
  * @return
  */
 public abstract boolean supportsLimitOffset();
 /**
  * 是否支持limit
  * @return
  */
 public abstract boolean supportsLimit();
 /**
  * 獲取增加了分頁屬性之后的SQL
  * @param sql
  * @param offset
  * @param limit
  * @return
  */
 public abstract String getLimitString(String sql, int offset, int limit);
}

再而我們就以Oracle與Mysql數據庫的分頁技術作下分別的實現

MySQLDialect.java-Mysql分頁方言

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
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方言實現

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
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();
 }
}

對應的Mybatis插件攔截器實現如下,攔截StatementHandler#prepare(Connection con)創建SQL語句對象方法

PaginationInterceptor.java

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
@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 {
  // 直接獲取攔截的對象,其實現類RoutingStatementHandler
  StatementHandler statementHandler = (StatementHandler) invocation
   .getTarget();
  BoundSql boundSql = statementHandler.getBoundSql();
  // 獲取元對象,主要用于獲取statementHandler所關聯的對象及屬性
  MetaObject metaStatementHandler = MetaObject.forObject(
   statementHandler, new DefaultObjectFactory(),
   new DefaultObjectWrapperFactory());
  MappedStatement mappedStmt= (MappedStatement) metaStatementHandler
   .getValue("delegate.mappedStatement".intern());
  // 只對queryPagination()方法進行分頁操作
  if(mappedStmt.getId().indexOf("queryPagination")==-1){
   return invocation.proceed();
  }
  // 重新構造分頁的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();
 }
 // 攔截對象
 @Override
 public Object plugin(Object target) {
  return Plugin.wrap(target, this);
 }
 @Override
 public void setProperties(Properties properties) {
 }
}

Spring對應的xml配置可如下,以oracle分頁為例子

?
1
2
3
4
5
6
<!-- oracle方言配置,用于oracle的分頁 -->
 <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>

使用以上的代碼以及配置即可完成對oracle數據庫以及mysql數據庫的分頁操作。并且博主對其中的某個點作下解析

Mybatis#MetaObject-元數據對象解析

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

MetaObject#forObject()

代理對象均通過此靜態方法進入

?
1
2
3
4
5
6
7
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);
 }
 }

我們可以直接觀察其中的構造函數,玄機就在此處

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
private MetaObject(Object object, ObjectFactory objectFactory, ObjectWrapperFactory objectWrapperFactory) {
 this.originalObject = object;
 this.objectFactory = objectFactory;
 this.objectWrapperFactory = objectWrapperFactory;
 // 所有的屬性獲取均通過objectWrapper類來獲取,此處主要對所代理的object對象類型進行判斷
 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);
 }
 }

為了理解的更為滲透,我們繼續跟進,最后我們得知其會調用Reflector類的構造函數

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
private Reflector(Class<?> clazz) {
 type = clazz;
 // 獲取構造類
 addDefaultConstructor(clazz);
 // 獲取get方法集合
 addGetMethods(clazz);
 // 獲取set方法集合
 addSetMethods(clazz);
 // 獲取內部屬性集合
 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便可以遍歷代理被代理類的所關聯的所有屬性,就拿RoutingStatementHandler類來說,經過上述操作后其便可以訪問內部屬性delegate以及delegate的內部屬性configuration/objectFactory/typeHandlerRegistry/resultSetHandler/parameterHandler/mappedStatement等屬性

MetaObject#getValue()

上述闡述的是如何代理被代理類的內部屬性,我們也簡單的看下是如何正確的調用

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
public Object getValue(String name) {
 // PropertyTokenizer與StringTokenizer類似,只是前者寫死以.為分隔符
 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字符串,可通過getValue("delegate.boundSql.sql")其中以.為分隔符并其中的屬性必須是內部屬性(區分大小寫)。

MetaObject#setValue()

原理同MetaObject#getValue()方法

總結

以上所述是小編給大家介紹的Spring Mybatis 分頁插件使用教程,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復大家的。在此也非常感謝大家對服務器之家網站的支持!

原文鏈接:https://www.cnblogs.com/question-sky/p/8462577.html

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 国产精品久久久久久模特 | 黄色毛片a级 | 精品久久久久久久久久久αⅴ | 国产精品一区视频 | 久久综合九色综合久久久精品综合 | 中文字幕22页 | 男人的天堂视频网站 | 欧美一区2区三区4区公司二百 | 99精品国产一区二区三区 | 看国产毛片 | 久草在线精品观看 | 久久艹艹艹| 久久欧美亚洲另类专区91大神 | 免费黄色一级网站 | www亚洲 | 欧美黄色一级片视频 | 97人人草 | 国产一区视频在线免费观看 | 亚洲成人久久精品 | 悠悠成人资源亚洲一区二区 | 国产精品免费视频观看 | 色综合激情| 亚洲视频综合网 | 精品中文字幕视频 | 久久午夜免费视频 | 成人免费淫片视频软件 | 圆产精品久久久久久久久久久 | 羞羞网站在线观看入口免费 | 国产精品成人久久久久a级 男女无遮挡羞羞视频 | 久久噜噜噜 | 成人午夜视频在线观看免费 | 国产精品久久久久国产精品三级 | 日夜操天天干 | 色淫视频 | 一级电影在线免费观看 | 91成人亚洲 | 久草最新 | 羞羞答答tv| 精品国产一区二区三区久久久蜜月 | 激情久久免费视频 | 精品久久久久久久久久久久久 |