日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 运维知识 > 数据库 >内容正文

数据库

ibatis mysql分页查询语句_简单实现ibatis的物理分页

發布時間:2023/12/15 数据库 34 豆豆
生活随笔 收集整理的這篇文章主要介紹了 ibatis mysql分页查询语句_简单实现ibatis的物理分页 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

一直以來ibatis的分頁都是通過滾動ResultSet實現的,應該算是邏輯分頁吧。邏輯分頁雖然能很干凈地獨立于特定數據庫,但效率在多數情況下不及特定數據庫支持的物理分頁,而hibernate的分頁則是直接組裝sql,充分利用了特定數據庫的分頁機制,效率相對較高。

網上已有《使ibatis支持hibernate式的物理分頁》等類似的文章以繼承SqlExecutor的方式實現了物理分頁,但是侵入性非常強,還得實現數據庫方言,方法非常復雜。同時SqlExecutor不是接口,對它的方法繼承也不能保證版本穩定。本文中將介紹的方式是實現queryWithSqlHandler方法,在查詢前將通過SqlHandler接口把sql傳給調用者,再用處理后的sql進行最終查詢,從而實現物理分頁等功能:

//用queryWithSqlHandler方法實現物理分頁的例子:

public Page queryPage(String statementId, param, final Page page){

final int pageNum = page.getPageNum();

final int pageSize = page.getPageSize();

List list = queryWithSqlHandler(statementId, param, new SqlHandler() {

@Override

public String handle(String sql, Object[] params) throws SQLException {

//查詢總記錄數

int total = getJdbcTemplate().queryForInt("select count(1) as RECORDS from (" + sql + ")", params);

page.setTotal(total);

//返回經過分頁包裝后的Sql

return "select * from (select row_.*, rownum row_num_ from ("+sql+") row_ where rownum<="+ pageNum*pageSize +") where row_num_ > "+ (pageNum-1)*pageSize;

}

});

page.setRows(list);

return page;

}

我們來看queryWithSqlHandler方法的實現:

/**

* 提供查詢前對sql處理的功能

*

* @param statementId

* @param param

* @param sqlHandler sql處理器

* @return

*/

private List queryWithSqlHandler(final String statementId, final Object param, final SqlHandler sqlHandler) {

final SqlMapClientImpl smc = getSqlMapClient();

if (sqlHandler != null) {

final MappedStatement mappedStatement = smc.getMappedStatement(statementId);

final Sql dySql = mappedStatement.getSql();

if (Proxy.isProxyClass(dySql.getClass())) {

log.debug("該Sql對象已經是代理對象,設置新的sql處理器。");

((SqlProxyHandler) Proxy.getInvocationHandler(dySql)).setSqlHandler(sqlHandler);

} else {

log.debug("創建Sql的代理對象!");

final SqlProxyHandler sqlProxyHandler = new SqlProxyHandler(dySql, sqlHandler);

final Class sqlClass = dySql.getClass();

final Sql proxy = (Sql) Proxy.newProxyInstance(sqlClass.getClassLoader(), sqlClass.getInterfaces(), sqlProxyHandler);

mappedStatement.setSql(proxy);

}

}

try {

return smc.queryForList(statementId, param);

} catch (SQLException ex) {

throw new RuntimeException("查詢失敗", ex);

}

}

private static final class SqlProxyHandler implements InvocationHandler {

private final Sql sql;

private final ThreadLocal sqlHandler = new ThreadLocal();

public SqlProxyHandler(Sql sql, SqlHandler handler) {

this.sql = sql;

setSqlHandler(handler);

}

public Sql getSql() {

return sql;

}

public void setSqlHandler(SqlHandler handler) {

this.sqlHandler.set(handler);

}

public SqlHandler getSqlHandler() {

return sqlHandler.get();

}

@Override

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

Object result = method.invoke(getSql(), args);

if ("getSql".equals(method.getName()) && getSqlHandler() != null) {

log.debug("原SQL: " + result);

final StatementScope statementScope = (StatementScope) args[0];

final Object[] params = statementScope.getParameterMap().getParameterObjectValues(statementScope, args[1]);

result = getSqlHandler().handle((String) result, params);

log.debug("處理后: " + result);

setSqlHandler(null);//執行完成后清除線程局部變量,下次調用需要設置新值,否則不攔截getSql方法

}

return result;

}

}

interface SqlHandler {

/**

* 處理sql語句

*

* @param sql ibatis生成的sql語句,其中參數用?號占位

* @param params sql對應的參數

* @return

* @throws Throwable

*/

String handle(String sql, Object[] params) throws Throwable;

}

總結

以上是生活随笔為你收集整理的ibatis mysql分页查询语句_简单实现ibatis的物理分页的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。