mysql 物理页_转mysql物理分页

packageplugin;importjava.sql.Connection;importjava.sql.PreparedStatement;importjava.sql.ResultSet;importjava.sql.SQLException;importjava.util.Properties;importorg.apache.ibatis.executor.parameter.ParameterHandler;importorg.apache.ibatis.executor.statement.StatementHandler;importorg.apache.ibatis.logging.Log;importorg.apache.ibatis.logging.LogFactory;importorg.apache.ibatis.mapping.BoundSql;importorg.apache.ibatis.mapping.MappedStatement;importorg.apache.ibatis.plugin.Interceptor;importorg.apache.ibatis.plugin.Intercepts;importorg.apache.ibatis.plugin.Invocation;importorg.apache.ibatis.plugin.Plugin;importorg.apache.ibatis.plugin.Signature;importorg.apache.ibatis.reflection.MetaObject;importorg.apache.ibatis.reflection.factory.DefaultObjectFactory;importorg.apache.ibatis.reflection.factory.ObjectFactory;importorg.apache.ibatis.reflection.wrapper.DefaultObjectWrapperFactory;importorg.apache.ibatis.reflection.wrapper.ObjectWrapperFactory;importorg.apache.ibatis.scripting.defaults.DefaultParameterHandler;importorg.apache.ibatis.session.Configuration;importorg.apache.ibatis.session.RowBounds;/*** 通过拦截StatementHandlerprepare方法,重写sql语句实现物理分页。

* 老规矩,签名里要拦截的类型只能是接口。**/@Intercepts({@Signature(type= StatementHandler.class, method = "prepare", args = {Connection.class})})public class PaginationInterceptor implementsInterceptor {private static final Log logger = LogFactory.getLog(PaginationInterceptor.class);private static final ObjectFactory DEFAULT_OBJECT_FACTORY = newDefaultObjectFactory();private static final ObjectWrapperFactory DEFAULT_OBJECT_WRAPPER_FACTORY = newDefaultObjectWrapperFactory();private static String DEFAULT_PAGE_SQL_ID = ".*Page$"; //需要拦截的ID(正则匹配)

@Overridepublic Object intercept(Invocation invocation) throwsThrowable {

StatementHandler statementHandler=(StatementHandler) invocation.getTarget();

MetaObject metaStatementHandler=MetaObject.forObject(statementHandler, DEFAULT_OBJECT_FACTORY,

DEFAULT_OBJECT_WRAPPER_FACTORY);

RowBounds rowBounds= (RowBounds) metaStatementHandler.getValue("delegate.rowBounds");//分离代理对象链(由于目标类可能被多个拦截器拦截,从而形成多次代理,通过下面的两次循环可以分离出最原始的的目标类)

while (metaStatementHandler.hasGetter("h")) {

Object object= metaStatementHandler.getValue("h");

metaStatementHandler=MetaObject.forObject(object, DEFAULT_OBJECT_FACTORY, DEFAULT_OBJECT_WRAPPER_FACTORY);

}//分离最后一个代理对象的目标类

while (metaStatementHandler.hasGetter("target")) {

Object object= metaStatementHandler.getValue("target");

metaStatementHandler=MetaObject.forObject(object, DEFAULT_OBJECT_FACTORY, DEFAULT_OBJECT_WRAPPER_FACTORY);

}//property在mybatis settings文件内配置

Configuration configuration = (Configuration) metaStatementHandler.getValue("delegate.configuration");//设置pageSqlId

String pageSqlId = configuration.getVariables().getProperty("pageSqlId");if (null == pageSqlId || "".equals(pageSqlId)) {

logger.warn("Property pageSqlId is not setted,use default '.*Page$' ");

pageSqlId=DEFAULT_PAGE_SQL_ID;

}

MappedStatement mappedStatement=(MappedStatement)

metaStatementHandler.getValue("delegate.mappedStatement");//只重写需要分页的sql语句。通过MappedStatement的ID匹配,默认重写以Page结尾的MappedStatement的sql

if(mappedStatement.getId().matches(pageSqlId)) {

BoundSql boundSql= (BoundSql) metaStatementHandler.getValue("delegate.boundSql");

Object parameterObject=boundSql.getParameterObject();if (parameterObject == null) {throw new NullPointerException("parameterObject is null!");

}else{

String sql=boundSql.getSql();//重写sql

String pageSql = sql + " LIMIT " + rowBounds.getOffset() + "," +rowBounds.getLimit();

metaStatementHandler.setValue("delegate.boundSql.sql", pageSql);//采用物理分页后,就不需要mybatis的内存分页了,所以重置下面的两个参数

metaStatementHandler.setValue("delegate.rowBounds.offset", RowBounds.NO_ROW_OFFSET);

metaStatementHandler.setValue("delegate.rowBounds.limit", RowBounds.NO_ROW_LIMIT);

}

}//将执行权交给下一个拦截器

returninvocation.proceed();

}

@OverridepublicObject plugin(Object target) {//当目标类是StatementHandler类型时,才包装目标类,否者直接返回目标本身,减少目标被代理的次数

if (target instanceofStatementHandler) {return Plugin.wrap(target, this);

}else{returntarget;

}

}

@Overridepublic voidsetProperties(Properties properties) {//To change body of implemented methods use File | Settings | File Templates.

}

}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值