`
aokunsang
  • 浏览: 815733 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

MyBatis的简单易用的分页研究

 
阅读更多

  网上很多Mybatis的分页文档,但是感觉千篇一律,并且不完整,只查询出来分页后的List列表,未封装查询的总条数(还需要单独查询记录总数)。
本文借鉴iteye网友对记录条数的封装思路,做了一个完整版的分页查询。

参考:http://stamen.iteye.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();
	}
}

 


 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);
	}

 

  • 大小: 63.5 KB
分享到:
评论

相关推荐

    基于SpringBoot与Mybatis的简单易用分页查询设计源码

    该项目是一款基于SpringBoot和Mybatis构建的简单易用分页查询解决方案源码,包含44个文件,涵盖31个Java源文件、3个Markdown文档、2个Kotlin源文件、1个Git忽略文件、1个许可证文件、1个属性文件、1个Gradle构建文件...

    MyBatis自动分页实现

    * 简洁易用:无需手动编写分页逻辑,MyBatis 将自动实现分页功能。 * 高效灵活:可以根据需要选择不同的拦截器,例如 MybatisPageInterceptor 和 MybatisSpringPageInterceptor。 * 可扩展性强:可以根据需要添加新...

    mybatis+springmvc分页

    最后,关于emp表的增删改操作,Mybatis同样提供了简单易用的API。例如,增加一条记录可以用`insert`标签,删除可以通过`delete`,更新则是`update`。在编写这些操作时,确保与数据库的实体类(如Emp)匹配,并正确...

    springmvc+mybatis+easyUI分页后台代码

    本项目是基于SpringMVC、MyBatis和EasyUI这三大框架实现的后台分页功能,旨在提供一个高效、易用的解决方案。下面将详细介绍这三个技术以及它们如何协同工作来实现前端页面的分页。 首先,SpringMVC是Spring框架的...

    最新Mybatis-PageHelper分页插件Jar

    总的来说,Mybatis-PageHelper是一个强大且易用的分页工具,能够有效地提升开发效率,减少代码量,同时保持良好的可读性和可维护性。通过理解其工作原理和正确配置,开发者可以轻松地在项目中实现高效分页。

    mybatis的pagahelper分页jar包

    MyBatis PagHelper 分页插件是针对MyBatis框架的一个高效、易用的分页插件,它能够简化在开发过程中对数据进行分页的操作。这个压缩包包含的"mybatis的pagahelper分页jar包"是 PagHelper 的核心组件,用于集成到你的...

    MyBatis 分页

    总的来说,MyBatis的分页机制是其强大功能的一部分,它通过简单易用的API和灵活的插件支持,使得数据库分页变得轻松而高效。无论是在小型项目还是大型企业级应用中,MyBatis都能胜任数据分页的需求,提升应用的性能...

    基于SpringMVC Mybatis框架实现分页

    在本项目中,我们主要探讨如何在SpringMVC和Mybatis框架中实现分页功能,同时结合RESTful API设计原则来构建一个高效、易用的Web应用。分页是大型Web应用不可或缺的一部分,它能够有效地管理和展示大量数据,提高...

    mybatis分页插件代码

    - **简单易用**:MyBatis-Plus提供了简单的API,使得开发者能快速上手。 - **自动分页**:通过BaseMapper接口,自动处理分页逻辑。 - **条件构造**:通过QueryWrapper或LambdaQueryWrapper,可以方便地构造复杂...

    Mybatis-plus分页插件的使用

    Mybatis-plus分页插件是Mybatis-Plus框架的一个强大特性,它为数据库查询提供了便捷的分页功能,尤其在处理大数据量的列表时显得尤为重要。分页不仅可以提高应用程序的性能,还可以提供更好的用户体验,避免一次性...

    mybatis 分页拦截器及拦截器配置

    这两个都是广受欢迎的MyBatis分页插件,它们提供了简单易用的API和配置,能够快速集成到项目中。以PageHelper为例,其核心配置包括以下几个步骤: 1. **添加依赖**:在项目的pom.xml文件中,添加PageHelper的依赖库...

    mybatis的分页插件

    4. **简单易用**:只需在 MyBatis 的配置文件中引入 PageHelper,并进行简单的配置,然后在 Mapper 接口中添加 @Select 和 @ResultMap 注解,即可实现分页查询。 5. **多数据源支持**:在多数据源环境中,...

    MyBatis的PageHelper分页插件

    1. **简单易用**:PageHelper只需要在项目中引入依赖,配置简单的参数即可快速使用,无需编写复杂的分页逻辑代码。 2. **高性能**:PageHelper内部采用了高效的分页算法,避免了全表扫描,提高了分页查询的性能。 3....

    mybatis分页插件的使用

    1. **简单易用**:开发者只需通过简单的API调用就能实现分页查询,无需关心底层的具体实现细节。 2. **兼容性好**:支持多种数据库,能够适应不同的开发环境。 3. **性能优化**:内置了高效的Count查询机制,减少了...

    10-2资料-spring+mybatis+pagehelper分页.rar

    PageHelper通过拦截器的方式,实现了简单易用的API,使得在查询数据时,只需添加几个参数,即可实现高效的分页效果,极大地提高了开发效率。 **Spring与MyBatis整合** Spring与MyBatis的整合主要是通过Spring的...

    springboot整合mybatis集成分页查询事务配置,页面模板

    在本项目中,我们主要探讨的是如何在SpringBoot框架下整合MyBatis,实现数据库的分页查询,事务管理,并配置前端页面模板。下面将详细阐述这些知识点。 首先,SpringBoot是一个快速开发框架,它简化了Spring应用的...

    好用的JAVA WEB分页中间件,比mybatis内置的分页好用.zip

    1. **易用性**:通过简单的API调用,即可实现复杂的分页逻辑,无需手动编写分页SQL。 2. **灵活性**:支持多种排序方式,包括字段排序和自定义SQL排序。 3. **高性能**:通过动态SQL,PageHelper在数据库层面进行...

    mybatis+bootstarp+pageHelper实现分页.rar

    通过这种方式,MyBatis、Bootstrap和PageHelper三者结合,可以在后端实现高效、灵活的分页处理,同时在前端提供直观易用的分页展示,提升用户体验。理解并熟练掌握这一技术栈对于开发大规模数据的Web应用至关重要。

    mybatis 分页插件jar包

    它的主要作用在于将原本繁琐的分页代码简化,通过简单的配置和调用,就能实现数据库查询结果的分页显示。这极大地提高了开发效率,减少了出错的可能性,使得开发者能够更加专注于业务逻辑的实现。 PageHelper的工作...

Global site tag (gtag.js) - Google Analytics