mybatis mysql 方言_iBatis3基于方言(Dialect)的分页

2023-11-10 20:00

本文主要是介绍mybatis mysql 方言_iBatis3基于方言(Dialect)的分页,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

(注:以下代码是基于ibatis3 beta4的扩展,ibatis3正式版如果实现改变,将会继续跟进修改)

iBatis3默认使用的分页是基于游标的分页,而这种分页在不同的数据库上性能差异不一致,最好的办法当然是使用类似hibernate的基于方言(Dialect)的物理分页功能。

iBatis3现在提供插件功能,通过插件我们可以编写自己的拦截器来拦截iBatis3的主要执行方法来完成相关功能的扩展。

能够拦截的的类如下:

Java代码

863dee2894cf8ef5449110aaefe28d6a.png

Executor

(update,query,flushStatements,commit,rollback,getTransaction,close,isClosed)

ParameterHandler

(getParameterObject,setParameters)

ResultSetHandler

(handleResultSets,handleOutputParameters)

StatementHandler

(prepare,parameterize,batch,update,query)

但此次我们感兴趣的是Executor.query()方法,查询方法最终都是执行该方法。

该方法完整是:

Java代码

863dee2894cf8ef5449110aaefe28d6a.png

Executor.query(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler) throws SQLException;

分页方言的基本实现:

以Mysql数据库示例,即有一个方法 query(sql,offset,limit);

Java代码

863dee2894cf8ef5449110aaefe28d6a.png

query("select * from user",5,10);

经过我们的拦截器拦截处理,变成

Java代码

863dee2894cf8ef5449110aaefe28d6a.png

query("select * from user limit 5,10",,);

1.拦截类实现:

Java代码

863dee2894cf8ef5449110aaefe28d6a.png

@Intercepts({@Signature(

type= Executor.class,

method = "query",

args = {MappedStatement.class, Object.class, RowBounds.class, ResultHandler.class})})

public class OffsetLimitInterceptor implements Interceptor{

static int MAPPED_STATEMENT_INDEX = ;

static int PARAMETER_INDEX = 1;

static int ROWBOUNDS_INDEX = 2;

static int RESULT_HANDLER_INDEX = 3;

Dialect dialect;

public Object intercept(Invocation invocation) throws Throwable {

processIntercept(invocation.getArgs());

return invocation.proceed();

}

void processIntercept(final Object[] queryArgs) {

//queryArgs = query(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler)

MappedStatement ms = (MappedStatement)queryArgs[MAPPED_STATEMENT_INDEX];

Object parameter = queryArgs[PARAMETER_INDEX];

final RowBounds rowBounds = (RowBounds)queryArgs[ROWBOUNDS_INDEX];

int offset = rowBounds.getOffset();

int limit = rowBounds.getLimit();

if(dialect.supportsLimit() && (offset != RowBounds.NO_ROW_OFFSET || limit != RowBounds.NO_ROW_LIMIT)) {

BoundSql boundSql = ms.getBoundSql(parameter);

String sql = boundSql.getSql().trim();

if (dialect.supportsLimitOffset()) {

sql = dialect.getLimitString(sql, offset, limit);

offset = RowBounds.NO_ROW_OFFSET;

} else {

sql = dialect.getLimitString(sql, , limit);

}

limit = RowBounds.NO_ROW_LIMIT;

queryArgs[ROWBOUNDS_INDEX] = new RowBounds(offset,limit);

BoundSql newBoundSql = new BoundSql(sql, boundSql.getParameterMappings(), boundSql.getParameterObject());

MappedStatement newMs = copyFromMappedStatement(ms, new BoundSqlSqlSource(newBoundSql));

queryArgs[MAPPED_STATEMENT_INDEX] = newMs;

}

}

private MappedStatement copyFromMappedStatement(MappedStatement ms,SqlSource newSqlSource) {

Builder builder = new MappedStatement.Builder(ms.getConfiguration(),ms.getId(),newSqlSource,ms.getSqlCommandType());

builder.resource(ms.getResource());

builder.fetchSize(ms.getFetchSize());

builder.statementType(ms.getStatementType());

builder.keyGenerator(ms.getKeyGenerator());

builder.keyProperty(ms.getKeyProperty());

builder.timeout(ms.getTimeout());

builder.parameterMap(ms.getParameterMap());

builder.resultMaps(ms.getResultMaps());

builder.cache(ms.getCache());

MappedStatement newMs = builder.build();

return newMs;

}

public Object plugin(Object target) {

return Plugin.wrap(target, this);

}

public void setProperties(Properties properties) {

String dialectClass = new PropertiesHelper(properties).getRequiredString("dialectClass");

try {

dialect = (Dialect)Class.forName(dialectClass).newInstance();

} catch (Exception e) {

throw new RuntimeException("cannot create dialect instance by dialectClass:"+dialectClass,e);

}

System.out.println(OffsetLimitInterceptor.class.getSimpleName()+".dialect="+dialectClass);

}

public static class BoundSqlSqlSource implements SqlSource {

BoundSql boundSql;

public BoundSqlSqlSource(BoundSql boundSql) {

this.boundSql = boundSql;

}

public BoundSql getBoundSql(Object parameterObject) {

return boundSql;

}

}

}

2.ibatis3配置文件内容:

Xml代码

863dee2894cf8ef5449110aaefe28d6a.png

3.MysqlDialect实现

Java代码

863dee2894cf8ef5449110aaefe28d6a.png

public class MySQLDialect extends Dialect{

public boolean supportsLimitOffset(){

return true;

}

public boolean supportsLimit() {

return true;

}

public String getLimitString(String sql, int offset, int limit) {

return getLimitString(sql,offset,String.valueOf(offset),limit,String.valueOf(limit));

}

public String getLimitString(String sql, int offset,String offsetPlaceholder, int limit, String limitPlaceholder) {

if (offset > ) {

return sql + " limit "+offsetPlaceholder+","+limitPlaceholder;

} else {

return sql + " limit "+limitPlaceholder;

}

}

}

这篇关于mybatis mysql 方言_iBatis3基于方言(Dialect)的分页的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/384929

相关文章

mybatis映射器配置小结

《mybatis映射器配置小结》本文详解MyBatis映射器配置,重点讲解字段映射的三种解决方案(别名、自动驼峰映射、resultMap),文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定... 目录select中字段的映射问题使用SQL语句中的别名功能使用mapUnderscoreToCame

Linux下MySQL数据库定时备份脚本与Crontab配置教学

《Linux下MySQL数据库定时备份脚本与Crontab配置教学》在生产环境中,数据库是核心资产之一,定期备份数据库可以有效防止意外数据丢失,本文将分享一份MySQL定时备份脚本,并讲解如何通过cr... 目录备份脚本详解脚本功能说明授权与可执行权限使用 Crontab 定时执行编辑 Crontab添加定

mybatis-plus如何根据任意字段saveOrUpdateBatch

《mybatis-plus如何根据任意字段saveOrUpdateBatch》MyBatisPlussaveOrUpdateBatch默认按主键判断操作类型,若需按其他唯一字段(如agentId、pe... 目录使用场景方法源码方法改造首先在service层定义接口service层接口实现总结使用场景my

MyBatis ParameterHandler的具体使用

《MyBatisParameterHandler的具体使用》本文主要介绍了MyBatisParameterHandler的具体使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参... 目录一、概述二、源码1 关键属性2.setParameters3.TypeHandler1.TypeHa

MyBatis-plus处理存储json数据过程

《MyBatis-plus处理存储json数据过程》文章介绍MyBatis-Plus3.4.21处理对象与集合的差异:对象可用内置Handler配合autoResultMap,集合需自定义处理器继承F... 目录1、如果是对象2、如果需要转换的是List集合总结对象和集合分两种情况处理,目前我用的MP的版本

MySQL中On duplicate key update的实现示例

《MySQL中Onduplicatekeyupdate的实现示例》ONDUPLICATEKEYUPDATE是一种MySQL的语法,它在插入新数据时,如果遇到唯一键冲突,则会执行更新操作,而不是抛... 目录1/ ON DUPLICATE KEY UPDATE的简介2/ ON DUPLICATE KEY UP

MySQL分库分表的实践示例

《MySQL分库分表的实践示例》MySQL分库分表适用于数据量大或并发压力高的场景,核心技术包括水平/垂直分片和分库,需应对分布式事务、跨库查询等挑战,通过中间件和解决方案实现,最佳实践为合理策略、备... 目录一、分库分表的触发条件1.1 数据量阈值1.2 并发压力二、分库分表的核心技术模块2.1 水平分

Python与MySQL实现数据库实时同步的详细步骤

《Python与MySQL实现数据库实时同步的详细步骤》在日常开发中,数据同步是一项常见的需求,本篇文章将使用Python和MySQL来实现数据库实时同步,我们将围绕数据变更捕获、数据处理和数据写入这... 目录前言摘要概述:数据同步方案1. 基本思路2. mysql Binlog 简介实现步骤与代码示例1

使用shardingsphere实现mysql数据库分片方式

《使用shardingsphere实现mysql数据库分片方式》本文介绍如何使用ShardingSphere-JDBC在SpringBoot中实现MySQL水平分库,涵盖分片策略、路由算法及零侵入配置... 目录一、ShardingSphere 简介1.1 对比1.2 核心概念1.3 Sharding-Sp

MySQL 表空却 ibd 文件过大的问题及解决方法

《MySQL表空却ibd文件过大的问题及解决方法》本文给大家介绍MySQL表空却ibd文件过大的问题及解决方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考... 目录一、问题背景:表空却 “吃满” 磁盘的怪事二、问题复现:一步步编程还原异常场景1. 准备测试源表与数据