Mybatis最入門---分頁查詢(攔截器分頁原理及實現)

前文,我們演示了物理分頁的Sql實現方式,這種方式使得我們每次在編寫查詢服務時,不斷的重複造輪子。這樣的代碼實現方式就顯得十分的笨拙了。本文是Mybatis分頁查詢的最後一片內容,我們將介紹基於攔截器的,精巧的實現方式。在閱讀這篇文章之前,強烈建議各位看官能夠先閱讀上文。這樣就能對下文我們提及的各種對象及他們之間的關係有一個清晰的關係。好了,廢話不多講,開始我們的正文部分吧。

準備工作:

a.操作系統 :win7 x64

b.基本軟件:MySQL,Mybatis,SQLyog

-------------------------------------------------------------------------------------------------------------------------------------

【本文只作爲原理分析及簡單分頁功能實現。實際生產環節,建議各位讀者選擇已經廣泛應用的第三方jar包或平臺性支持的實現】

-------------------------------------------------------------------------------------------------------------------------------------

1.創建本文我們將使用的工程Mybatis12,工程結構圖如下:【重點文件我們給出,其他配置文件請讀者參考前文工程】


2.管理分頁的對象PagePOJO的具體內容如下:【這裏我們給出基本分頁的示例,更多需求請讀者自行完成】

[java] view plain copy
  1. package com.csdn.ingo.entity;  
  2. /** 
  3. *@author 作者 E-mail:ingo 
  4. *@version 創建時間:2016年4月27日下午6:27:05 
  5. *類說明 
  6. */  
  7. public class PagePOJO {  
  8.     private int totalNumber;//當前表中總條目數量  
  9.     private int currentPage;//當前頁的位置  
  10.     private int totalPage;//總頁數  
  11.     private int pageSize;//頁面大小  
  12.     private int startIndex;//檢索的起始位置  
  13.     private int totalSelect;//檢索的總數目  
  14.     //...省略其他set,get方法  
  15.     public void setTotalNumber(int totalNumber) {  
  16.         this.totalNumber = totalNumber;  
  17.         this.count();  
  18.     }  
  19.     //...省略其他set,get方法  
  20.       
  21.     public PagePOJO(int totalNumber, int currentPage, int totalPage, int pageSize, int startIndex, int totalSelect) {  
  22.         super();  
  23.         this.totalNumber = totalNumber;  
  24.         this.currentPage = currentPage;  
  25.         this.totalPage = totalPage;  
  26.         this.pageSize = pageSize;  
  27.         this.startIndex = startIndex;  
  28.         this.totalSelect = totalSelect;  
  29.     }  
  30.     public void count(){  
  31.         int totalPageTemp = this.totalNumber/this.pageSize;  
  32.         int plus = (this.totalNumber%this.pageSize)==0?0:1;  
  33.         totalPageTemp = totalPageTemp+plus;  
  34.         if(totalPageTemp<=0){  
  35.             totalPageTemp=1;  
  36.         }  
  37.         this.totalPage = totalPageTemp;//總頁數  
  38.           
  39.         if(this.totalPage<this.currentPage){  
  40.             this.currentPage = this.totalPage;  
  41.         }  
  42.         if(this.currentPage<1){  
  43.             this.currentPage=1;  
  44.         }  
  45.         this.startIndex = (this.currentPage-1)*this.pageSize;//起始位置等於之前所有頁面輸乘以頁面大小  
  46.         this.totalSelect = this.pageSize;//檢索數量等於頁面大小  
  47.     }  
  48. }  
3.新增單元測試方法,如下:【測試數據讀者可以自行更換】

[java] view plain copy
  1. @Test  
  2.     public void testSelectPage() {  
  3.         try {  
  4.             // 創建分頁對象  
  5.             //(int totalNumber, int currentPage, int totalPage, int pageSize, int startIndex, int totalSelect)  
  6.             PagePOJO page = new PagePOJO(511314);  
  7.             Map<String,Object> params = new  HashMap<String,Object>();  
  8.             params.put("page", page);  
  9.             UserInfoDao userInfo = sqlSession.getMapper(UserInfoDao.class);  
  10.             List<UserInfo> re = userInfo.selectByPage(params);  
  11.             System.out.println(re);  
  12.         } catch (Exception e) {  
  13.             e.printStackTrace();  
  14.         }  
  15.     }  

-------------------------------------------------------------------------------------------------------------------------------------

上面與分頁有關的,比較簡單的內容我們先提供給大家。下面我們來詳細的解釋,攔截器實現分頁的原理,各位看官睜大眼睛啊!

-------------------------------------------------------------------------------------------------------------------------------------

分頁開始之前的問題:

  1. Mybatis如何找到我們新增的攔截服務。
  2. 自定義的攔截服務應該在什麼時間攔截查詢動作。即什麼時間截斷Mybatis執行流。
  3. 自定義的攔截服務應該攔截什麼樣的對象。不能攔截什麼樣的對象。
  4. 自定義的攔截服務攔截的對象應該具有什麼動作才能被攔截。
  5. 自定義的攔截服務如何獲取上下文中傳入的參數信息。
  6. 如何把簡單查詢,神不知鬼不覺的,無侵入性的替換爲分頁查詢語句。
  7. 最後,攔截器應該如何交還被截斷的Mybatis執行流。

帶着這些問題,我們來看看我們自定義的攔截服務是如何實現的。

-------------------------------------------------------------------------------------------------------------------------------------------------------

1.首先,我們看看查詢語句的時序圖,如下:


2.回顧一下,Mybatis允許我們能夠進行切入的點,如下:【上文的運行細節圖中也有描述,具體內容請參考前文。】


【老外其實從命名上已經給我們了充足的信息,我們結合時序圖已經能夠猜到其大概的含義及執行時間,
鑑於篇幅的關係,具體的功能及作用,請參考前文詳細內容。】

【結論】從上面的1.2兩步,我們知道分頁攔截的合理時機是在StatementHandler中。

3.現在,來看看這個StatementHandler的具體內容,如下:

【成員方法】


【類間關係】


這裏請讀者回顧上文中講述的查詢執行流,即這裏將會執行prepare(Connection)方法,如下:

[java] view plain copy
  1. @Override  
  2.   public Statement prepare(Connection connection) throws SQLException {  
  3.     ErrorContext.instance().sql(boundSql.getSql());  
  4.     Statement statement = null;  
  5.     try {  
  6.       statement = instantiateStatement(connection);  
  7.       setStatementTimeout(statement);  
  8.       setFetchSize(statement);  
  9.       return statement;  
  10.     } catch (SQLException e) {  
  11.       closeStatement(statement);  
  12.       throw e;  
  13.     } catch (Exception e) {  
  14.       closeStatement(statement);  
  15.       throw new ExecutorException("Error preparing statement.  Cause: " + e, e);  
  16.     }  
  17.   }  

執行到instantiateStatement(connection);如下:【PreparedStatementHandler中】

[java] view plain copy
  1. @Override  
  2.  protected Statement instantiateStatement(Connection connection) throws SQLException {  
  3.    String sql = boundSql.getSql();  
  4.    if (mappedStatement.getKeyGenerator() instanceof Jdbc3KeyGenerator) {  
  5.      String[] keyColumnNames = mappedStatement.getKeyColumns();  
  6.      if (keyColumnNames == null) {  
  7.        return connection.prepareStatement(sql, PreparedStatement.RETURN_GENERATED_KEYS);  
  8.      } else {  
  9.        return connection.prepareStatement(sql, keyColumnNames);  
  10.      }  
  11.    } else if (mappedStatement.getResultSetType() != null) {  
  12.      return connection.prepareStatement(sql, mappedStatement.getResultSetType().getValue(), ResultSet.CONCUR_READ_ONLY);  
  13.    } else {  
  14.      return connection.prepareStatement(sql);  
  15.    }  
  16.  }  

【重點!!!】【這裏我們就發現真正執行查詢語句的地方connection.prepareStatement(...)】

由此,就解釋了爲什麼在StatementHandler中進行攔截。

4.於是,我們可以得到攔截器的對應註解內容如下:【繼承自Mybatis的Interceptor,必須實現3個方法,如下:】

[java] view plain copy
  1. @Intercepts({@Signature(type=StatementHandler.class,method="prepare",args={Connection.class})})  

5.現在,自定義的攔截器的基本結構如下:

[java] view plain copy
  1. package com.csdn.ingo.interceptor;  
  2.   
  3.   
  4. /** 
  5. *@author 作者 E-mail:ingo 
  6. *@version 創建時間:2016年4月27日下午6:55:09 
  7. */  
  8. @Intercepts({@Signature(type=StatementHandler.class,method="prepare",args={Connection.class})})  
  9. public class PageInterceptor implements Interceptor{  
  10.       
  11.     /* (non-Javadoc) 
  12.      * 攔截器要執行的方法 
  13.      */  
  14.     public Object intercept(Invocation invocation) throws Throwable {  
  15.         //...  
  16.     }  
  17.   
  18.     /* (non-Javadoc) 
  19.      * 攔截器需要攔截的對象 
  20.      */  
  21.     public Object plugin(Object target) {  
  22.         //...  
  23.     }  
  24.   
  25.     /* (non-Javadoc) 
  26.      * 設置初始化的屬性值 
  27.      */  
  28.     public void setProperties(Properties properties) {  
  29.         //...  
  30.     }  
  31.   
  32. }  
6.我們先看看攔截器上面基本結構中需要攔截的對象的具體方法實現:

[java] view plain copy
  1. /* (non-Javadoc) 
  2.      * 攔截器需要攔截的對象,target。this,當前類的實例 
  3.      */  
  4.     public Object plugin(Object target) {  
  5.         return Plugin.wrap(target, this);  
  6.     }  

接下來,深入的看看wrap方法,如下:

[java] view plain copy
  1. public static Object wrap(Object target, Interceptor interceptor) {  
  2.     Map<Class<?>, Set<Method>> signatureMap = getSignatureMap(interceptor);  
  3.     Class<?> type = target.getClass();  
  4.     Class<?>[] interfaces = getAllInterfaces(type, signatureMap);  
  5.     if (interfaces.length > 0) {  
  6.       return Proxy.newProxyInstance(  
  7.           type.getClassLoader(),  
  8.           interfaces,  
  9.           new Plugin(target, interceptor, signatureMap));  
  10.     }  
  11.     return target;  
  12.   }  
接着,再看看getSignatureMap(interceptor);的詳細內容,如下:

[java] view plain copy
  1. private static Map<Class<?>, Set<Method>> getSignatureMap(Interceptor interceptor) {  
  2.     Intercepts interceptsAnnotation = interceptor.getClass().getAnnotation(Intercepts.class);  
  3.     // ...  
  4.      
  5.     return signatureMap;  
  6.   }  
由此看到:annotation註解等其他相關內容都放在Map中返回。最後,判斷出是否生成代理對象。

-------------------------------------------------------------------------------------------------------------------------------------

7.現在,我們已經成功的攔截到了目標對象,然後,就開始要改變查詢過程了。在這裏,出現了兩個基本的問題

  • 如何獲取原始的查詢語句,即mapper文件中的sql語句。
  • 如何獲取分頁信息。

其實,在方法intercept(Invocation invocation)的參數中,已經包含了StatementHandler的信息,我們需要做的就是取出其中的信息,操作步驟如下:

a.先取出StatementHandler,如下:

[java] view plain copy
  1. StatementHandler statementHandler = (StatementHandler)invocation.getTarget();  
·關於StatementHandler其有兩個實現,【見上文】,默認情況下,程序會執行BaseStatementHandler下的內容。如下:



其中的mappedStatement中就包含着我們需要找的sql信息。如下:【如果讀者已經看過前面的配置詳解部分的內容,已經不會感到陌生】


現在,我們已經找到了我們想要的內容的具體位置,那我們應該怎樣取出來呢?此時應該有看官大喊一聲:這還不簡單!可是我們來看看這部分的源碼內容,如下:

[java] view plain copy
  1. protected final MappedStatement mappedStatement;  
【注意】

其在BaseStatementHandler中是protected的,我們在沒有繼承等關係的條件下,是無法直接取出來的。

因此,我們就需要引入一個Mybatis已經實現了的對象:MetaObject。關於這個對象,可以先暫時的理解爲幫助我們獲取或設置該對象的原本不可訪問的屬性。

在3.3.1版本中,其內部細節如下:


這裏,我們先只用到了public static MetaObject forObject(.....) ,其他內容有興趣的讀者可以自行學習。

於是,我們操作的對象就由StatementHandler的實例,變爲MetaObject的實例,如下:

[java] view plain copy
  1. StatementHandler statementHandler = (StatementHandler)invocation.getTarget();  
  2. MetaObject metaObject = MetaObject.forObject(statementHandler, SystemMetaObject.DEFAULT_OBJECT_FACTORY, SystemMetaObject.DEFAULT_OBJECT_WRAPPER_FACTORY,new DefaultReflectorFactory());  
  3.           
現在對象已經獲取了,就要獲取其中的值了,在具體操作之前,我們再來梳理一下這裏的類間關係,如下:【這裏需要閱讀過前文程序執行流程,或者之前已有了解】

現在,我們終於可以去獲取其屬性值了,獲取的方法爲參數名,OGNL表達式。具體內容如下:

[java] view plain copy
  1. MappedStatement mappedStatement = (MappedStatement) metaObject.getValue("delegate.mappedStatement");  
  2. String id = mappedStatement.getId();  

8.獲得目標對象之後,現在要做進一步判斷,即,當前被代理的對象是普通查詢,還是分頁查詢。這裏具體方式有很多,我們給出一種簡單實現,更多實現及用法就行讀者自行嘗試吧。

如,我們在這裏定義,id中(即mapper文件中sql語句的唯一id)以ByPage字符串結尾的,我們就認爲該sql語句按照分頁插敘來執行。具體的代碼試下如下:

[java] view plain copy
  1. if(id.matches(".+ByPage$")){  
  2.      //....  
  3. }  

-------------------------------------------------------------------------------------------------------------------------------------

9.通過上面一系列的步驟,我們已經成功的捕獲了目標對象。接下來,我們就要開始加入我們“神不知鬼不覺‘”的代碼了。

【具體分爲下面的幾個步驟】

  • 獲取原始sql語句
  • 執行滿足一定條件的查詢,查詢出結果總數,用於計算分頁頁面總數
  • 獲取將dao層傳入的分頁參數
  • 將獲取分頁查詢的參數,用於重新拼裝分頁查詢語句。
  • 查詢結束之後,交還程序執行流,退出攔截器

這裏仍然需要前面執行流程的知識,默認情況下,執行的是PreparedStatementHandler。因此,我們可以從PreparedStatementHandler中尋找需要的sql語句。具體見上文第3步。這裏僅給出省略內容,如下:

[java] view plain copy
  1. @Override  
  2.   protected Statement instantiateStatement(Connection connection) throws SQLException {  
  3.     String sql = boundSql.getSql();  
  4.     //...  
  5.   }  
由此,我們發現Sql語句隱藏在boundSql中。回頭看看,已經獲得了的StatementHandler,其具體內容我們也在上文截圖給大家。

於是,就得到了下面的代碼:

[java] view plain copy
  1. BoundSql boundSql = statementHandler.getBoundSql();  
  2. Map<String,Object> params = (Map<String,Object>)boundSql.getParameterObject();  

現在,各位看官可以尖叫了!!!所有的材料已經準備齊了。

-------------------------------------------------------------------------------------------------------------------------------------

10.查詢總數對應的Mysql語句。【其他數據庫原理一致,讀者自行完成即可】

[java] view plain copy
  1. String sql = boundSql.getSql();  
  2. String countSql = "select count(*)from ("+sql+")a";  
【注意:這裏我們已經獲得了傳入參數與完整的sql語句,因此,select count還有更加高效的寫法,請讀者自行完成。】

11.接下來就是執行查詢總數的語句。其總數的值用於完成分頁頁面數量控制。執行方法如下:

【回顧我們的註解,args配置】

[java] view plain copy
  1. Connection connection = (Connection) invocation.getArgs()[0];  
  2. //利用原始sql語句的方法執行  
  3. PreparedStatement countStatement = connection.prepareStatement(countSql);  
  4. //在本例中,查詢參數爲空,但實際應用時,多爲帶有條件的分頁查詢,下面的這句話就是獲取查詢條件的參數  
  5. ParameterHandler parameterHandler = (ParameterHandler) metaObject.getValue("delegate.parameterHandler");  
  6. //經過set方法,就可以正確的執行sql語句  
  7. parameterHandler.setParameters(countStatement);  
  8. ResultSet rs = countStatement.executeQuery();  
  9. //當結果集中有值時,表示頁面數量大於等於1  
  10. if(rs.next()){  
  11.     page.setTotalNumber(rs.getInt(1));  
  12. }  

到此,select count就執行完畢了。

11.分頁的查詢語句爲:【其他數據庫原理一致,讀者自行完成即可】

[java] view plain copy
  1. String pageSql = sql+" limit "+page.getStartIndex()+","+page.getTotalSelect();  

12.查詢總數,分頁查詢的語句已經準備完成,如何替換原來的查詢語句呢?再回頭看看MetaObject對象,剛纔,我們說它提供給我們獲取或設置該對象的原本不可訪問的屬性。因此,就來利用它實現替換sql語句的功能。如下:【備註:這句話具體解釋與上文getValue一致,請參考上文即可】

[java] view plain copy
  1. metaObject.setValue("delegate.boundSql.sql", pageSql);  
到此,我們成功的把原有的簡單查詢語句替換爲分頁查詢語句了,現在是時候將程序的控制權交還給Mybatis了,具體代碼如下:

[java] view plain copy
  1. return invocation.proceed();  
13.最後一步,自定義的攔截器需要交給Mybatis管理,這樣才能使得Mybatis的執行與攔截器的執行結合在一起,即,攔截器需要註冊到mybatis-config配置文件中。具體內容如下:

[html] view plain copy
  1. <?xml version="1.0" encoding="UTF-8" ?>  
  2. <!DOCTYPE configuration  
  3. PUBLIC "-//mybatis.org//DTD Config 3.0//EN"  
  4. "http://mybatis.org/dtd/mybatis-3-config.dtd">  
  5. <configuration>  
  6.     <properties resource="jdbc.properties"/>  
  7.     <settings>  
  8.         <setting name="logImpl" value="LOG4J"/>  
  9.     </settings>  
  10.     <typeAliases>  
  11.         <package name="com.csdn.ingo.entity"/>  
  12.     </typeAliases>  
  13.     <plugins>  
  14.         <plugin interceptor="com.csdn.ingo.interceptor.PageInterceptor"></plugin>  
  15.     </plugins>  
  16.     <environments default="development">  
  17.         <environment id="development">  
  18.             <transactionManager type="JDBC" />  
  19.             <dataSource type="POOLED">  
  20.                 <property name="driver" value="${jdbc.driverClassName}" />  
  21.                 <property name="url" value="${jdbc.url}" />  
  22.                 <property name="username" value="${jdbc.username}" />  
  23.                 <property name="password" value="${jdbc.password}" />  
  24.             </dataSource>  
  25.         </environment>  
  26.     </environments>  
  27.     <mappers>  
  28.           <mapper resource="mappers/UserInfoMapper.xml"/>  
  29.     </mappers>  
  30. </configuration>  
----------------------------------------------------------------------------------------------------------------------------------

上面是所有的原理內容,我們再給出完整的代碼供大家學習

----------------------------------------------------------------------------------------------------------------------------------
14.PageInterceptor的完整內容如下:

[java] view plain copy
  1. package com.csdn.ingo.interceptor;  
  2.   
  3. import java.sql.Connection;  
  4. import java.sql.PreparedStatement;  
  5. import java.sql.ResultSet;  
  6. import java.util.Map;  
  7. import java.util.Properties;  
  8.   
  9. import org.apache.ibatis.executor.parameter.ParameterHandler;  
  10. import org.apache.ibatis.executor.statement.StatementHandler;  
  11. import org.apache.ibatis.mapping.BoundSql;  
  12. import org.apache.ibatis.mapping.MappedStatement;  
  13. import org.apache.ibatis.plugin.Interceptor;  
  14. import org.apache.ibatis.plugin.Intercepts;  
  15. import org.apache.ibatis.plugin.Invocation;  
  16. import org.apache.ibatis.plugin.Plugin;  
  17. import org.apache.ibatis.plugin.Signature;  
  18. import org.apache.ibatis.reflection.DefaultReflectorFactory;  
  19. import org.apache.ibatis.reflection.MetaObject;  
  20. import org.apache.ibatis.reflection.ReflectorFactory;  
  21. import org.apache.ibatis.reflection.SystemMetaObject;  
  22.   
  23. import com.csdn.ingo.entity.PagePOJO;  
  24.   
  25. /** 
  26. *@author 作者 E-mail:ingo 
  27. *@version 創建時間:2016年4月27日下午6:55:09 
  28. *類說明 
  29. */  
  30. @Intercepts({@Signature(type=StatementHandler.class,method="prepare",args={Connection.class})})  
  31. public class PageInterceptor implements Interceptor{  
  32.       
  33.     /* (non-Javadoc) 
  34.      * 攔截器要執行的方法 
  35.      */  
  36.     public Object intercept(Invocation invocation) throws Throwable {  
  37.         StatementHandler statementHandler = (StatementHandler)invocation.getTarget();  
  38.         MetaObject metaObject = MetaObject.forObject(statementHandler, SystemMetaObject.DEFAULT_OBJECT_FACTORY, SystemMetaObject.DEFAULT_OBJECT_WRAPPER_FACTORY,new DefaultReflectorFactory());  
  39.         MappedStatement mappedStatement = (MappedStatement) metaObject.getValue("delegate.mappedStatement");  
  40.         String id = mappedStatement.getId();  
  41.         if(id.matches(".+ByPage$")){  
  42.               
  43.             BoundSql boundSql = statementHandler.getBoundSql();  
  44.             Map<String,Object> params = (Map<String,Object>)boundSql.getParameterObject();  
  45.             PagePOJO page = (PagePOJO)params.get("page");  
  46.             String sql = boundSql.getSql();  
  47.             String countSql = "select count(*)from ("+sql+")a";  
  48.             Connection connection = (Connection) invocation.getArgs()[0];  
  49.             PreparedStatement countStatement = connection.prepareStatement(countSql);  
  50.             ParameterHandler parameterHandler = (ParameterHandler) metaObject.getValue("delegate.parameterHandler");  
  51.             parameterHandler.setParameters(countStatement);  
  52.             ResultSet rs = countStatement.executeQuery();  
  53.             if(rs.next()){  
  54.                 page.setTotalNumber(rs.getInt(1));  
  55.             }  
  56.             String pageSql = sql+" limit "+page.getStartIndex()+","+page.getTotalSelect();  
  57.             metaObject.setValue("delegate.boundSql.sql", pageSql);  
  58.         }  
  59.         return invocation.proceed();  
  60.     }  
  61.   
  62.     /* (non-Javadoc) 
  63.      * 攔截器需要攔截的對象 
  64.      */  
  65.     public Object plugin(Object target) {  
  66.         return Plugin.wrap(target, this);  
  67.     }  
  68.   
  69.     /* (non-Javadoc) 
  70.      * 設置初始化的屬性值 
  71.      */  
  72.     public void setProperties(Properties properties) {  
  73.           
  74.     }  
  75. }  
15.Mapper.xml文件中的sql語句如下:

[html] view plain copy
  1. <select id="selectByPage" parameterType="Map" resultMap="UserInfoResult">  
  2.     select * from userinfo   
  3. </select>  
【注意:】

這裏的sql語句就是簡單查詢

傳入參數爲Map,供多參,條件查詢的等場景使用。【最好這麼使用,因爲攔截器只有1個】

16.執行單元測試方法,看看查詢結果,如下:




如果看到類似上文輸出,表明已經成功執行了分頁查詢!掌聲!!!

----------------------------------------------------------------------------------------------------------------------------------

至此, Mybatis最入門---分頁查詢(攔截器分頁原理及實現)結束

備註:

1.本文僅作爲原理解釋,實際應用時,建議各位看官最好使用更加嚴謹的第三方jar包,或者平臺性的支持。

2.上文未給出的代碼,在前文中均有給出,請讀者自行查閱前文。


發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章