设为首页 收藏本站
查看: 780|回复: 0

[经验分享] MyBatis的简单易用的分页研究

[复制链接]

尚未签到

发表于 2016-11-26 08:37:04 | 显示全部楼层 |阅读模式
  网上很多Mybatis的分页文档,但是感觉千篇一律,并且不完整,只查询出来分页后的List列表,未封装查询的总条数(还需要单独查询记录总数)。
本文借鉴iteye网友对记录条数的封装思路,做了一个完整版的分页查询。
  参考:http://stamen.iyunv.com/blog/1901576
  使用的是和Hibernate一样方言的模式来实现各种数据库的分页,先看基础类Dialect.java:

/**
* @description 方言基类
* @author aokunsang
* @date 2013-8-23
*/
public abstract class Dialect {
protected final static String SELECT = "select";
protected final static String SELECT_DISTINCT = "select distinct";
public final static String COUNT_FIELD = "TOTAL";
/* 数据库类型 */
public static enum DatabaseType{
MYSQL,ORACLE,SQLSERVER
}
/**
* 获取分页SQL语句
* @param sql The SQL statement to base the limit query off of.
* @param offset Offset of the first row to be returned by the query (zero-based)
* @param limit Maximum number of rows to be returned by the query
* @return A new SQL statement with the LIMIT clause applied.
*/
public abstract String getLimitString(String sql, int offset, int limit);
/**
* 获取总条数的SQL语句[该方法通用]
* @param sql
* @return
*/
public String getCountString(String sql){
StringBuilder sb = new StringBuilder(sql.trim());
String querySqlLowered = sql.trim().toLowerCase();
int orderByIndex = querySqlLowered.toLowerCase().indexOf("order by");
if (orderByIndex > 0) {
sb.delete(orderByIndex, orderByIndex + sql.substring(orderByIndex).length());
}
String querySqlFrom = sb.toString().toLowerCase();
int fromIndex = querySqlFrom.indexOf("from");
if(fromIndex > 0){
sb.delete(0, fromIndex);
}
sb.insert(0, "select count(1) as " + COUNT_FIELD + " ");
return sb.toString();
}
}

  再看OracleDialect.java,其他的数据库分页参考hibernate核心包的源码org.hibernate.dialect包名下的类(附图)。

/**
* @description Oracle方言
* @author aokunsang
* @date 2013-8-23
*/
public class OracleDialect extends Dialect {
/* (non-Javadoc)
* @see com.aokunsang.mybatis.study.util.Dialect#getLimitString(java.lang.String, int, int)
*/
@Override
public String getLimitString(String sql, int offset, int limit) {
sql = sql.trim();
boolean isForUpdate = false;
if(sql.toLowerCase().endsWith(" for update")){
sql = sql.substring( 0, sql.length()-11 );
isForUpdate = true;
}
StringBuffer pagingSelect = new StringBuffer( sql.length()+100 );
if (offset > 0) {  //不是第一页
pagingSelect.append("select * from ( select row_.*, rownum rownum_ from ( ");
}else {
pagingSelect.append("select * from ( ");
}
pagingSelect.append(sql);
if (offset > 0) {
pagingSelect.append(" ) row_ where rownum <= "+(limit+offset)+") where rownum_ > "+offset);
}else {
pagingSelect.append(" ) where rownum <= "+limit);
}
if(isForUpdate) {
pagingSelect.append( " for update" );
}
return pagingSelect.toString();
}
}
DSC0000.jpg
 Mybatis的分页拦截器基本和网上查询到的一样,没做什么修改,参考类PaginationInterceptor.java:


/**
* @description 分页拦截器
* @author aokunsang
* @date 2013-8-23
*/
@Intercepts(
{@Signature(
type = StatementHandler.class,
method = "prepare",
args = {Connection.class}
)
})
public class PaginationInterceptor implements Interceptor {
private static final Log logger = LogFactory.getLog(PaginationInterceptor.class);
private static final String DATABASE_TYPE_ID = "dialect";
private static final String PAGE_SQL_MATCH_ID = "pageSqlId";
private static final DatabaseType DEFAULT_DATA_SOURCE_TYPE = DatabaseType.valueOf("ORACLE");  //默认数据库类型
//必须以Page结尾的查询ID  [解译:配置需要拦截的Mapper.xml中的查询ID(正则匹配),如:<select id="xx"></select> 中的xx]
private static final String DEFAULT_PAGE_SQL_ID = ".*Page$";
private static final ObjectFactory DEFAULT_OBJECT_FACTORY = new DefaultObjectFactory();
private static final ObjectWrapperFactory DEFAULT_OBJECT_WRAPPER_FACTORY = new DefaultObjectWrapperFactory();
@Override
public Object intercept(Invocation invocation) throws Throwable {
StatementHandler statementHandler = (StatementHandler)invocation.getTarget();
MetaObject metaStatementHandler = MetaObject.forObject(statementHandler,DEFAULT_OBJECT_FACTORY,DEFAULT_OBJECT_WRAPPER_FACTORY);
Connection conn = (Connection)invocation.getArgs()[0];
BoundSql boundSql = statementHandler.getBoundSql();
RowBounds rowBounds = (RowBounds)metaStatementHandler.getValue("delegate.rowBounds");
if(rowBounds == null || rowBounds == RowBounds.DEFAULT){
return invocation.proceed();
}
Configuration configuration = (Configuration)metaStatementHandler.getValue("delegate.configuration");
DatabaseType databaseType = null;
try {
databaseType = DatabaseType.valueOf(configuration.getVariables().getProperty(DATABASE_TYPE_ID).toUpperCase());
} catch (Exception e) {
databaseType = DEFAULT_DATA_SOURCE_TYPE;
logger.warn(String.format("在configurationProperties中未找到数据库类型设置,默认使用[%s]。", "ORACLE"));
}
String pageSqlId = "";
try {
pageSqlId = configuration.getVariables().getProperty(PAGE_SQL_MATCH_ID);
} catch (Exception e) {
pageSqlId = DEFAULT_PAGE_SQL_ID;
logger.warn(String.format("在configurationProperties中未找到SQL查询语句ID的匹配设置,默认使用[%s]。",DEFAULT_PAGE_SQL_ID));
}
Dialect dialect = null;
switch(databaseType){
case ORACLE:{
dialect = new OracleDialect();break;
}
case MYSQL:{}
}
MappedStatement mappedStatement = (MappedStatement)metaStatementHandler.getValue("delegate.mappedStatement");
if(mappedStatement.getId().matches(pageSqlId)){
String originalSql = boundSql.getSql();
//获取记录总行数,并将其绑定到当前线程中
MybatisPageHelper.buildPageCount(conn, mappedStatement, boundSql, dialect.getCountString(originalSql));
//设置分页
metaStatementHandler.setValue("delegate.boundSql.sql", dialect.getLimitString(originalSql, rowBounds.getOffset(), rowBounds.getLimit()) );   
//采用物理分页后,就不需要mybatis的内存分页了,所以重置下面的两个参数
metaStatementHandler.setValue("delegate.rowBounds.offset", RowBounds.NO_ROW_OFFSET);   
metaStatementHandler.setValue("delegate.rowBounds.limit", RowBounds.NO_ROW_LIMIT);
}
return invocation.proceed();
}
@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) {}
}

  怎么保存和获取记录总条数的,是MybatisPageHelper.java类操作的:

/**
* @description Mybatis分页帮助类(主要是设置和获取记录总数)
* @author aokunsang
* @date 2013-8-26
*/
public final class MybatisPageHelper {
private static final Log logger = LogFactory.getLog(MybatisPageHelper.class);  
/**
* 保存计算总行数的值
*/  
private static ThreadLocal<Integer> totalRowCountHolder = new ThreadLocal<Integer>();
/**
* 获取记录总数
* @param conn
* @param mappedStatement
* @param boundSql
* @param countSql   查询记录总数SQL语句
* @return
*/
static void buildPageCount(Connection conn,MappedStatement mappedStatement,BoundSql boundSql,String countSql){
int count = 0;
ResultSet rs = null;
PreparedStatement ps = null;
try {
ps = conn.prepareStatement(countSql);
BoundSql bound_count_sql = new BoundSql(mappedStatement.getConfiguration(),
countSql,
boundSql.getParameterMappings(),
boundSql.getParameterObject());
ParameterHandler pmHandler = new DefaultParameterHandler(mappedStatement, boundSql.getParameterObject(), bound_count_sql);
pmHandler.setParameters(ps);
rs = ps.executeQuery();
if(rs.next()){
count = rs.getInt(Dialect.COUNT_FIELD);
}
if (logger.isDebugEnabled()) {  
logger.debug("Total count: {"+count+"}");
}  
totalRowCountHolder.set(count);
} catch (SQLException e) {
e.printStackTrace();
} finally{
try {
rs.close();
} catch (SQLException e) { e.printStackTrace();}
try {
ps.close();
} catch (SQLException e) { e.printStackTrace();}
}
}
/**
* 获取当前线程对应的分页查询的总行数
* @return
*/  
public static int getTotalRowCount() {  
return totalRowCountHolder.get();  
}
/**
* 组装RowBounds
* @param pageNumber     当前页码
* @param pageSize       查询记录数
* @return
*/
public static RowBounds toRowBounds(int pageNumber,int pageSize){
int offset = (pageNumber-1) * pageSize;  //根据页码数和展示记录数,计算出偏移量
return new RowBounds(offset, pageSize);
}

  接下来看配置文件,因为使用的是Spring整合的Mybatis,并且没有使用Mybatis-config.xml,spring的配置文件如下:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"  
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"   
xmlns:aop="http://www.springframework.org/schema/aop"  
xmlns:tx="http://www.springframework.org/schema/tx"  
xmlns:context="http://www.springframework.org/schema/context"  
xsi:schemaLocation="   
http://www.springframework.org/schema/beans   
http://www.springframework.org/schema/beans/spring-beans-3.1.xsd   
http://www.springframework.org/schema/tx   
http://www.springframework.org/schema/tx/spring-tx-3.1.xsd  
http://www.springframework.org/schema/context   
http://www.springframework.org/schema/context/spring-context-3.1.xsd   
http://www.springframework.org/schema/aop   
http://www.springframework.org/schema/aop/spring-aop-3.1.xsd" default-autowire="byName">
<context:property-placeholder location="classpath:config/database.properties"/>
<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
<property name="driverClass" value="${datasource.driverClassName}"></property>
<property name="jdbcUrl" value="${datasource.url}"></property>
<property name="user" value="${datasource.username}"></property>
<property name="password" value="${datasource.password}"></property>
<property name="acquireIncrement" value="${c3p0.acquireIncrement}"></property>
<property name="initialPoolSize" value="${c3p0.initialPoolSize}"></property>
<property name="minPoolSize" value="${c3p0.minPoolSize}"></property>
<property name="maxPoolSize" value="${c3p0.maxPoolSize}"></property>
<property name="maxIdleTime" value="${c3p0.maxIdleTime}"></property>
<property name="idleConnectionTestPeriod" value="${c3p0.idleConnectionTestPeriod}"></property>
<property name="maxStatements" value="${c3p0.maxStatements}"></property>
<property name="numHelperThreads" value="${c3p0.numHelperThreads}"></property>
<property name="acquireRetryAttempts" value="${c3p0.acquireRetryAttempts}"></property>
<property name="breakAfterAcquireFailure" value="${c3p0.breakAfterAcquireFailure}"></property>
<property name="testConnectionOnCheckout" value="${c3p0.testConnectionOnCheckout}"></property>
</bean>
<bean id="txManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource"></property>
</bean>
<bean id="sqlSessionFacroty" class="org.mybatis.spring.SqlSessionFactoryBean">
<property name="dataSource" ref="dataSource"></property>
<property name="mapperLocations" value="classpath*:com/aokunsang/mybatis/study/mapper/*.xml"></property>
<property name="typeAliasesPackage" value="com.aokunsang.mybatis.study.po"></property>
<property name="plugins" ref="pageInterceptor"></property>
<property name="configurationProperties">
<props>
<prop key="dialect">oracle</prop>
<prop key="pageSqlId">.*Page$</prop>
</props>
</property>
</bean>
<bean class="org.mybatis.spring.mapper.MapperScannerConfigurer">
<property name="basePackage" value="com.aokunsang.mybatis.study.mapper"></property>
</bean>
<!-- MyBatis的物理分页拦截器 -->
<bean id="pageInterceptor" class="com.aokunsang.mybatis.study.page.PaginationInterceptor"/>
<!-- 注解式事务管理 -->
<tx:annotation-driven transaction-manager="txManager"/>
<!-- 自动扫描组件,需要把controller去掉,否则影响事务管理 -->
<context:component-scan base-package="com.aokunsang.mybatis.study">
<context:exclude-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
</context:component-scan>
</beans>

  具体使用方式,先来看看Mapper接口和xml配置文件:

/**
* @description Mapper接口
* @author aokunsang
* @date 2013-8-9
*/
public interface ProjectGenreMapper {
public List<TbProjectGenre> findPrjGenrelkPage(@Param("namelk") String namelk, RowBounds rowBounds);
}

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.aokunsang.mybatis.study.mapper.ProjectGenreMapper">
<select id="findPrjGenrelkPage" resultType="TbProjectGenre">
select * from tb_project_genre
<where>
<if test="namelk!=null">
NAME like '%'||#{namelk}||'%'
</if>
</where>
order by ID desc
</select>
</mapper>
  再看Service实现类:

/**
* @description
* @author aokunsang
* @date 2013-8-9
*/
@Service
@Transactional
public class ProjectGenreServiceImpl implements IProjectGenreService {
@Autowired
private ProjectGenreMapper pgMapper;
@Override
public PaginationSupport<TbProjectGenre> findPrjGenrelkPage(String namelk,
int pageNumber, int pageSize) {
List<TbProjectGenre> list = pgMapper.findPrjGenrelkPage(namelk, MybatisPageHelper.toRowBounds(pageNumber, pageSize));
PaginationSupport<TbProjectGenre> pager = new PaginationSupport<TbProjectGenre>(list,pageNumber,pageSize);
pager.setTotalCount(MybatisPageHelper.getTotalRowCount());  //设置记录总数[这里是重点哦]
return pager;
}
}
  最后看看单元测试:

/**
* @description
* @author aokunsang
* @date 2013-8-9
*/
//@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations={"classpath:applicationContext-jdbc.xml"})
public class JunitTest extends AbstractJUnit4SpringContextTests{
static{
//动态加载log4j日志
PropertyConfigurator.configure(JunitTest.class.getClassLoader().getResource("log4j.properties"));
}
@Autowired
private IProjectGenreService pgService;
@Test
public void list(){
PaginationSupport<TbProjectGenre> pager = pgService.findPrjGenrelkPage("水", 1, 2);
System.out.println("------------>>>"+pager.getTotalCount()+"-------->>>"+pager.getItems().size());
}
}
  另外需要注意的是,想查看SQL语句,经过我测试,把log4j日志设置成如下即可:

# Output pattern : date [thread] priority category - message
log4j.rootLogger=DEBUG, Console
#Console
log4j.appender.Console=org.apache.log4j.ConsoleAppender
#正式环境时候 console级别设置为WARN
log4j.appender.Console.Threshold=DEBUG
log4j.appender.Console.layout=org.apache.log4j.PatternLayout
log4j.appender.Console.layout.ConversionPattern=%d{yyyy-MM-dd HH:mm:ss} [%t] %-5p [%c] - %m%n
# Spring Stuff
log4j.logger.org.springframework=WARN
# 这个是Mapper接口的包名(最重要)
log4j.logger.com.aokunsang.mybatis.study.mapper=DEBUG   
log4j.logger.com.mchange.v2=WARN
log4j.logger.org.apache.ibatis=WARN
log4j.logger.org.mybatis.spring=WARN
  再修改下,因为在service实现类中,所有的分页最后都会执行一段代码:

pager.setTotalCount(MybatisPageHelper.getTotalRowCount());  //设置记录总数
  这里修改为Spring AOP拦截相关方法,自动设置添加记录总数,看这里:

/**
* @description
* @author aokunsang
* @date 2013-8-26
*/
@Component
@Aspect
public class TotalRowValueMount {
@AfterReturning(value="execution(* com.aokunsang.mybatis.study.service.impl.*.*(..))",returning="pager")
@SuppressWarnings("unchecked")
public void setTotalCount(JoinPoint jp, PaginationSupport pager){
if(pager!=null){  //如果返回值为PaginationSupport,则执行
pager.setTotalCount(MybatisPageHelper.getTotalRowCount());
}
}
}
  然后需要在spring的配置文件中加上对aspectj注解的支持就行了。

<!-- 启动对@Asceptj注解的支持 -->
<aop:aspectj-autoproxy />
  此时,Service实现类的方法改为如下即可:

@Override
public PaginationSupport<TbProjectGenre> findPrjGenrelkPage(String namelk,
int pageNumber, int pageSize) {
List<TbProjectGenre> list = pgMapper.findPrjGenrelkPage(namelk, MybatisPageHelper.toRowBounds(pageNumber, pageSize));
return new PaginationSupport<TbProjectGenre>(list,pageNumber,pageSize);
}

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-305671-1-1.html 上篇帖子: mybatis @param 类型 转换错误(关键字) 下篇帖子: Spring集成MyBatis进行项目开发(一)
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表