天天看點

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方法的實作:

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 {

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

}