(五)SpringBoot配置mybatis分页拦截器

        用mybatis进行数据库物理分页查询,假如每次都手动设置分页信息,并手动编写分页sql将会很费工夫。而目前已有不少分页插件可供使用。其主要原理是拦截执行查询过程中sqlSession的查询,先进行数据条数查询,再查询分页数据,组成带分页信息的数据。

本教程并非分页拦截器的编写,而是SpringBoot分页拦截器的配置。具体的分页拦截器的编写可查看教程:

https://blog.csdn.net/lanhezhong/article/details/64907025 (Spring-mybatis3.4物理分页实现)

当然也可以用开源的分页插件实现,不过配置到SpringBoot中步骤是一样的。

 

配置SpringBoot分页插件步骤:

一、选择或编写好分页插件,下面是我自己编写的Mysql分页插件,需要3个类:

PaginationInterceptor.java

package com.lan.BootMybatis.interceptors;
 
import java.sql.Connection;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.Properties;

import org.apache.ibatis.executor.parameter.ParameterHandler;
import org.apache.ibatis.executor.statement.StatementHandler;
import org.apache.ibatis.mapping.BoundSql;
import org.apache.ibatis.mapping.MappedStatement;
import org.apache.ibatis.plugin.Interceptor;
import org.apache.ibatis.plugin.Intercepts;
import org.apache.ibatis.plugin.Invocation;
import org.apache.ibatis.plugin.Plugin;
import org.apache.ibatis.plugin.Signature;
import org.apache.ibatis.reflection.DefaultReflectorFactory;
import org.apache.ibatis.reflection.MetaObject;
import org.apache.ibatis.reflection.ReflectorFactory;
import org.apache.ibatis.reflection.factory.DefaultObjectFactory;
import org.apache.ibatis.reflection.factory.ObjectFactory;
import org.apache.ibatis.reflection.wrapper.DefaultObjectWrapperFactory;
import org.apache.ibatis.reflection.wrapper.ObjectWrapperFactory;
import org.apache.ibatis.scripting.defaults.DefaultParameterHandler;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.util.StringUtils;

import com.lan.BootMybatis.base.model.SimplePage;

/**
 * 分页拦截器mybatis
 * 
 * @author Lan
 * @date 2016年9月29日
 */
@Intercepts({ @Signature(type = StatementHandler.class, method = "prepare", args = { Connection.class , Integer.class}) })
public class PaginationInterceptor implements Interceptor{
 
	private static final Logger logger = LoggerFactory.getLogger(PaginationInterceptor.class);
 
	private static final ObjectFactory DEFAULT_OBJECT_FACTORY = new DefaultObjectFactory();
	private static final ObjectWrapperFactory DEFAULT_OBJECT_WRAPPER_FACTORY = new DefaultObjectWrapperFactory();
	private static final ReflectorFactory DEFAULT_REFLECTOR_FACTORY = new DefaultReflectorFactory();
	private static String dialect = "mysql";
 
	@Override
	public Object intercept(Invocation invocation) throws Throwable {
		// 获得拦截的对象
		StatementHandler statementHandler = (StatementHandler) invocation.getTarget();
		// 待执行的sql的包装对象
		BoundSql boundSql = statementHandler.getBoundSql();
		// 判断是否是查询语句
		if (isSelect(boundSql.getSql())) {
			// 获得参数集合
			Object params = boundSql.getParameterObject();
 
			if (params instanceof Map) { // 请求为多个参数,参数采用Map封装
				return complexParamsHandler(invocation, boundSql, (Map) params);
			} else if (params instanceof SimplePage) { // 单个参数且为Page,则表示该操作需要进行分页处理
				return simpleParamHandler(invocation, boundSql, (SimplePage) params);
			}
		}
		return invocation.proceed();
	}
	
	private Object complexParamsHandler(Invocation invocation, BoundSql boundSql, Map params) throws Throwable {
        //判断参数中是否指定分页
        if (containsPage(params)) {
            return pageHandlerExecutor(invocation, boundSql, (SimplePage) params.get("page"));
        } else {
            return invocation.proceed();
        }
    }
	
	private boolean containsPage(Map params) {
		if(params==null){
			return false;
		}else if(!params.containsKey("page")){
			return false;
		}
		Object page = params.get("page");
		if(page==null){
			return false;
		}else if(page instanceof SimplePage){
			return true;
		}
        return false;
    }
 
	private boolean isSelect(String sql) {
		if (!StringUtils.isEmpty(sql) && sql.toUpperCase().trim().startsWith("SELECT")) {
			return true;
		}
		return false;
	}
 
	private Object simpleParamHandler(Invocation invocation, BoundSql boundSql, SimplePage page) throws Throwable {
		return pageHandlerExecutor(invocation, boundSql, page);
	}
 
	private Object pageHandlerExecutor(Invocation invocation, BoundSql boundSql, SimplePage page) throws Throwable {
		// 获得数据库连接
		Connection connection = (Connection) invocation.getArgs()[0];
		// 使用Mybatis提供的MetaObject,该对象主要用于获取包装对象的属性值
		MetaObject statementHandler = MetaObject.forObject(invocation.getTarget(), DEFAULT_OBJECT_FACTORY,
				DEFAULT_OBJECT_WRAPPER_FACTORY, DEFAULT_REFLECTOR_FACTORY);
 
		// 获取该sql执行的结果集总数
		int maxSize = getTotalSize(connection, (MappedStatement) statementHandler.getValue("delegate.mappedStatement"),
				boundSql);
 
		// 生成分页sql
		page.setTotalRecord(maxSize);
		String wrapperSql = getPageSql(boundSql.getSql(), page);
 
		MetaObject boundSqlMeta = MetaObject.forObject(boundSql, DEFAULT_OBJECT_FACTORY, DEFAULT_OBJECT_WRAPPER_FACTORY,
				DEFAULT_REFLECTOR_FACTORY);
		// 修改boundSql的sql
		boundSqlMeta.setValue("sql", wrapperSql);
		return invocation.proceed();
	}
 
	private int getTotalSize(Connection connection, MappedStatement mappedStatement, BoundSql boundSql) {
		String countSql = getCountSql(boundSql.getSql());
		PreparedStatement countStmt;
		ResultSet rs;
		List closeableList = new ArrayList();
 
		try {
			countStmt = connection.prepareStatement(countSql);
			BoundSql countBS = new BoundSql(mappedStatement.getConfiguration(), countSql,
					boundSql.getParameterMappings(), boundSql.getParameterObject());
			setParameters(countStmt, mappedStatement, countBS, boundSql.getParameterObject());
			rs = countStmt.executeQuery();
 
			if (rs.next()) {
				return rs.getInt(1);
			}
			closeableList.add(countStmt);
			closeableList.add(rs);
		} catch (SQLException e) {
			logger.error("append an exception[{}] when execute sql[{}] with {}", e, countSql,
					boundSql.getParameterObject());
		} finally {
			for (AutoCloseable closeable : closeableList) {
				try {
					if (closeable != null)
						closeable.close();
				} catch (Exception e) {
					logger.error("append an exception[{}] when close resource[{}] ", e, closeable);
				}
			}
		}
		return 0;
	}
 
	private void setParameters(PreparedStatement ps, MappedStatement mappedStatement, BoundSql boundSql,
			Object parameterObject) throws SQLException {
		ParameterHandler parameterHandler = new DefaultParameterHandler(mappedStatement, parameterObject, boundSql);
		parameterHandler.setParameters(ps);
	}
 
	@Override
	public Object plugin(Object target) {
		// 当目标类是StatementHandler类型时,才包装目标类,否者直接返回目标本身,减少目标被代理的次数
        if (target instanceof StatementHandler) {
            return Plugin.wrap(target, this);
        } else {
            return target;
        }
	}
 
	@Override
	public void setProperties(Properties properties) {
		
	}
	
	public String getCountSql(String sql) {
		if("mysql".equals(dialect)){
			return "select count(0) from (" + sql + ") as total";
		}
		return sql;
	}
	
	public String getPageSql(String sql, SimplePage page) {
		if(page.getPage()<=0){
			page.setPage(1);
		}
		if(page.getRows()<=0){
			page.setRows(20);
		}
		int startRow = (page.getPage()-1)*page.getRows();
		
		if(startRow>=page.getTotalRecord()){
			page.setPage(1);
			startRow=0;
		}
		if("mysql".equals(dialect)){
			return sql+" limit "+startRow+", "+page.getRows();
		}
		return sql;
	}
}

SimplePage.java

package com.lan.BootMybatis.base.model;
 
import java.io.Serializable;
import java.util.List;
 
/**
 * 简单分页对象
 * @author Lan
 * @date 2016年9月29日
 */
public class SimplePage implements Serializable{
	
	protected static final long serialVersionUID = 5136213157391895517L;
	
	protected int page = 1;// 页码,默认是第一页
	protected int rows = 10;// 每页显示的记录数,默认是10
	protected int totalRecord;// 总记录数
	protected List data;// 对应的当前页记录
	
	
	public int getPage() {
		return page;
	}
	public void setPage(int page) {
		this.page = page;
	}
	public int getRows() {
		return rows;
	}
	public void setRows(int rows) {
		this.rows = rows;
	}
	public int getTotalRecord() {
		return totalRecord;
	}
	public void setTotalRecord(int totalRecord) {
		this.totalRecord = totalRecord;
	}
	public List getData() {
		return data;
	}
	public SimplePage setData(List data) {
		this.data = data;
		return this;
	}
 
}

Page.java

package com.lan.BootMybatis.base.model;
 
import java.util.List;
 
public class Page extends SimplePage{
	private static final long serialVersionUID = -6190845403265328029L;
	
	private boolean isFirstPage = true;//是否是第一页
	private boolean isLastPage = false;//是否是最后一页
	private int pageCount = 0;//当前页总记录数
	private int totalPage = 0;//总页数
	private int prePage = 1;//上一页页码
	private int nextPage = 1;//下一页页码
	
	public Page() {
		super();
	}
	
	public Page(int page, int rows) {
		super();
		setPage(page);
		setRows(rows);
	}
	
	@Override
	public Page setData(List data){
		super.setData(data);
		if(data!=null && data.size()>0){
			pageCount = data.size();
			if(this.page==1){
				isFirstPage=true;
			}else{
				isFirstPage=false;
			}
			//***
			totalPage = (int)Math.ceil(totalRecord/(double)rows);
			//***
			if(page==totalPage){
				isLastPage = true;
			}else{
				isLastPage = false;
			}
			//***
			if(isFirstPage){
				prePage = 1;
			}else{
				prePage = page-1;				
			}
			//***
			if(isLastPage){
				nextPage = 1;
			}else{
				nextPage = page+1;				
			}
		}else{
			isLastPage = true;
		}
		return this;
	}
	
	
	/**只要get方法**/
	public boolean isFirstPage() {
		return isFirstPage;
	}
 
	public boolean isLastPage() {
		return isLastPage;
	}
 
	public int getPageCount() {
		return pageCount;
	}
 
	public int getTotalPage() {
		return totalPage;
	}
 
	public int getPrePage() {
		return prePage;
	}
 
	public int getNextPage() {
		return nextPage;
	}
}

 

二、创建mybatis的配置文件mybatis-config.xml,其中配置着分页拦截器,我放在resources/mybatis/下

mybatis-config.xml




	
		
	
	
		
	

 

三、application.properties中配置mybatis配置文件mybatis-config.xml位置:

加这一行mybatis.config-location=classpath:mybatis/mybatis-config.xml

#mysql-setting
spring.datasource.url=jdbc\:mysql\://localhost\:3306/my_db
spring.datasource.username=root
spring.datasource.password=root
spring.datasource.driver-class-name=com.mysql.jdbc.Driver
spring.datasource.type=com.alibaba.druid.pool.DruidDataSource
spring.datasource.initialSize=5
spring.datasource.minIdle=5
spring.datasource.maxActive=20

#mybatis
mybatis.type-aliases-package=com.lan.BootMybatis.model
mybatis.mapper-locations=classpath:mybatis/mapper/*.xml
mybatis.config-location=classpath:mybatis/mybatis-config.xml

 

到此就完成了分页拦截器的配置,主要是第二、第三步。接着就可以根据分页插件要求,编写测试代码了。

测试:

TestController.java:

@RestController
public class TestController {

	@Resource
	private IDemoService demoService;
	
	@RequestMapping("findPage")
	public Page findPage(Page page){
		return demoService.findPage(page);
	}
}

DemoServiceImpl.java (DemoService.java接口略)

@Service("demoService")
public class DemoServiceImpl implements IDemoService{

	@Resource
	private DemoMapper demoMapper;
	
	@Override
	public Page findPage(Page page) {
		List list = demoMapper.findPage(page);
		page.setData(list);
		return page;
	}
}

DemoMapper.java

public interface DemoMapper {
	//@Param("page")是应分页插件要求编写的
	List findPage(@Param("page") Page page);
}

DemoMapper.xml

启动后访问:http://127.0.0.1:8080/findPage?rows=2

(五)SpringBoot配置mybatis分页拦截器_第1张图片

访问:http://127.0.0.1:8080/findPage?rows=5

(五)SpringBoot配置mybatis分页拦截器_第2张图片

 

总结:文章篇幅太长太啰嗦,主要是看第二、第三步。

如有需要,可从GitHub下载demo项目尝试:

https://github.com/heesonlan/SpringBoot-MybatisPagination.git

demo项目结构如下:

(五)SpringBoot配置mybatis分页拦截器_第3张图片

 

author:蓝何忠

email:[email protected]

你可能感兴趣的:(SpringBoot)