mysql尾页查询_mysql 大表分页查询 翻页 优化方案

本文探讨了MySQL分页查询的效率问题,尤其是随着页数增加查询时间变长。提出了一种优化策略,通过转换offset并利用上一次查询的记录来改善查询效率。针对id排序连续和不连续的情况,以及根据其他字段如创建时间排序的情况,提供了不同的解决方案,并给出了相应的Java代码示例。
摘要由CSDN通过智能技术生成

mysql分页查询是先查询出来offset+limit行数据,然后放弃前offset,取limit条记录,造成了越往后的页数,查询时间越长

一般优化思路是转换offset,让offset尽可能的小,最好能每次查询都是第一页,也就是offset为0

查询按id排序的情况

一、如果查询是根据id排序的,并且id是连续的

这种网上介绍比较多,根据要查的页数直接算出来id的范围

比如offset=40, limit=10, 表示查询第5页数据,那么第5页开始的id是41,增加查询条件:id>40  limit 10

二、如果查询是根据id排序的,但是id不是连续的

通常翻页页数跳转都不会很大,那我们可以根据上一次查询的记录,算出来下一次分页查询对应的新的 offset和 limit,也就是离上一次查询记录的offset

分页查询一般会有两个参数:offset和limit,limit一般是固定,假设limit=10

那为了优化offset太大的情况,每次查询需要提供两个额外的参数

参数lastEndId: 上一次查询的最后一条记录的id

参数lastEndOffset: 上一次查询的最后一条记录对应的offset,也就是上一次查询的offset+limit

第一种情况(与第二种其实是一样):跳转到下一页,增加查询条件:id>lastEndId limit 10

第二种情况:往下翻页,跳转到下任意页,算出新的newOffset=offset-lastEndOffset,增加查询条件:id>lastEndId offset newOffset limit 10,但是如果newOffset也还是很大,比如,直接从第一页跳转到最后一页,这时候我们可以根据id逆序(如果原来id是正序的换成倒序,如果是倒序就换成正序)查询,根据总数量算出逆序查询对应的offset和limit,那么 newOffset = totalCount - offset - limit, 查询条件:id=totalCount ,也就是算出来的newOffset 可能小于0, 所以最后一页的newOffset=0,limit = totalCount - offset

第三种情况:往上翻页,跳转到上任意页,根据id逆序 ,newOffset = lastEndOffset- offset - limit-1, 查询条件:id

补充的按id排序的后端实现代码,前端和第三种类似,就不补充了

/**

* 如果是根据id排序的分页,根据上一次查询的记录的id优化分页查询

*

* @see 将会自动添加id排序

* @param lastEndId

* 上一次查询的最后一条记录的id

* @param lastEndOffset

* 上一次查询的最后一条记录对应的偏移量 offset+limit

**/

public Page page(QueryBuilder queryBuilder, Integer lastEndId, Integer lastEndOffset, int offset, int limit) {

FromBuilder fromBuilder = queryBuilder.from(getModelClass());

Page page = new Page<>();

int count = dao.count(fromBuilder);

page.setTotal(count);

if (count == 0) {

return page;

}

if (offset == 0 || lastEndId == null || lastEndOffset == null) {

List list = dao

.find(SelectBuilder.selectFrom(fromBuilder.offsetLimit(offset, limit).order().desc("id").end()));

page.setData(list);

return page;

}

boolean isForward = offset >= lastEndOffset;

if (isForward) {

// offset增大的情况

int calcOffset = offset - lastEndOffset;

int calcOffsetFormEnd = count - offset - limit;

if (calcOffsetFormEnd <= calcOffset) {

// 离尾页近的情况,从尾页开始算

isForward = false;

if (calcOffsetFormEnd > 0) {

fromBuilder.order().asc("id").end().offsetLimit(calcOffsetFormEnd, limit);

} else {

// 最后一页

fromBuilder.order().asc("id").end().offsetLimit(0, calcOffsetFormEnd + limit);

}

} else {

fromBuilder.where().andLt("id", lastEndId).end().order().desc("id").end().offsetLimit(calcOffset,

limit);

}

} else {

// offset减少的情况

int calcOffset = lastEndOffset - offset - limit - 1;

if (calcOffset < offset) {

fromBuilder.where().andGt("id", lastEndId).end().order().asc("id").end().offsetLimit(calcOffset, limit);

} else {

// 离首页近的情况,直接查询

fromBuilder.offsetLimit(offset, limit).order().desc("id").end();

isForward = true;

}

}

List list = dao.find(SelectBuilder.selectFrom(fromBuilder));

if (!isForward) {

list.sort(new Comparator() {

@Override

public int compare(T o1, T o2) {

return o1.getId() < o2.getId() ? 1 : -1;

}

});

}

page.setData(list);

return page;

}

三,如果查询是根据其他字段,比如一般使用的创建时间(createTime)排序

这种跟第二种情况差不多,区别是createTime不是唯一的,所以不能确定上一次最后一条记录对应的创建时间,哪些是下一页的,哪些是上一页的

这时候,增加一个请求参数lastEndCount:表示上一次查询最后一条记录对应的创建时间,有多少条是这同一时间的,这个根据上一次的数据统计

根据第二种情况下计算出来的newOffset加上lastEndCount,就是新的offset,其他的处理方式和第二种一致

java 示例:

/**

* 如果是根据创建时间排序的分页,根据上一条记录的创建时间优化分布查询

*

* @see 将会自动添加createTime排序

* @param lastEndCreateTime

* 上一次查询的最后一条记录的创建时间

* @param lastEndCount 上一次查询的时间为lastEndCreateTime的数量

* @param lastEndOffset 上一次查询的最后一条记录对应的偏移量 offset+limit

**/

public Page page(QueryBuilder queryBuilder, Date lastEndCreateTime, Integer lastEndCount, Integer lastEndOffset,

int offset, int limit) {

FromBuilder fromBuilder = queryBuilder.from(getModelClass());

Page page = new Page<>();

int count = dao.count(fromBuilder);

page.setTotal(count);

if (count == 0) {

return page;

}

if (offset == 0 || lastEndCreateTime == null || lastEndCount == null || lastEndOffset == null) {

List list = dao.find(

SelectBuilder.selectFrom(fromBuilder.offsetLimit(offset, limit).order().desc("createTime").end()));

page.setData(list);

return page;

}

boolean isForward = offset >= lastEndOffset;

if (isForward) {

int calcOffset = offset - lastEndOffset + lastEndCount;

int calcOffsetFormEnd = count - offset - limit;

if (calcOffsetFormEnd <= calcOffset) {

isForward = false;

if (calcOffsetFormEnd > 0) {

fromBuilder.order().asc("createTime").end().offsetLimit(calcOffsetFormEnd, limit);

} else {

fromBuilder.order().asc("createTime").end().offsetLimit(0, calcOffsetFormEnd + limit);

}

} else {

fromBuilder.where().andLe("createTime", lastEndCreateTime).end().order().desc("createTime").end()

.offsetLimit(calcOffset, limit);

}

} else {

//这里没有判断是否离首页近,可以加上优化查询,直接按正常查询

fromBuilder.where().andGe("createTime", lastEndCreateTime).end().order().asc("createTime").end()

.offsetLimit(lastEndOffset - offset - limit - 1 + lastEndCount, limit);

}

List list = dao.find(SelectBuilder.selectFrom(fromBuilder));

if (!isForward) {

list.sort(new Comparator() {

@Override

public int compare(T o1, T o2) {

return o1.getCreateTime().before(o2.getCreateTime()) ? 1 : -1;

}

});

}

page.setData(list);

return page;

}

前端js参数,基于bootstrap table

this.lastEndCreateTime = null;

this.currentEndCreateTime = null;

this.currentDataLength = null;

this.isRefresh = false;

this.currentEndOffset = 0;

this.lastEndOffset = 0;

this.lastEndCount = 0;

this.currentEndCount = 0;

$("#" + this.tableId).bootstrapTable({

url: url,

method: 'get',

contentType: "application/x-www-form-urlencoded",//请求数据内容格式 默认是 application/json 自己根据格式自行服务端处理

dataType:"json",

dataField:"data",

pagination: true,

sidePagination: "server", // 服务端请求

pageList: [10, 25, 50, 100, 200],

search: true,

showRefresh: true,

toolbar: "#" + tableId + "Toolbar",

iconSize: "outline",

icons: {

refresh: "icon fa-refresh",

},

queryParams: function(params){

if(params.offset == 0){

this.currentEndOffset = params.offset + params.limit;

}else{

if(params.offset + params.limit==this.currentEndOffset){

//刷新

this.isRefresh = true;

params.lastEndCreateTime = this.lastEndCreateTime;

params.lastEndOffset = this.lastEndOffset;

params.lastEndCount = this.lastEndCount;

}else{

console.log(this.currentEndCount);

//跳页

this.isRefresh = false;

params.lastEndCreateTime = this.currentEndCreateTime;

params.lastEndOffset = this.currentEndOffset;

params.lastEndCount = this.currentEndCount;

if(this.currentDataLength != null && this.currentDataLength != params.limit){

//最后一页数据不满limit,调整lastEndOffset的值

params.lastEndOffset = this.currentEndOffset-

params.limit+this.currentDataLength;

}

this.lastEndOffset = this.currentEndOffset;

this.currentEndOffset = params.offset + params.limit;

console.log(params.lastEndOffset+","+params.lastEndCreateTime);

}

}

return params;

},

onSearch: function (text) {

this.keyword = text;

},

onPostBody : onPostBody,

onLoadSuccess: function (resp) {

if(resp.code!=0){

alertUtils.error(resp.msg);

}

var data = resp.data;

var dateLength = data.length;

this.currentDataLength = dateLength;

if(dateLength==0){

return;

}

if(!this.isRefresh){

this.lastEndCreateTime = this.currentEndCreateTime;

this.currentEndCreateTime = data[data.length-1].createTime;

this.lastEndCount = this.currentEndCount;

this.currentEndCount = 0;

for (var i = 0; i < resp.data.length; i++) {

var item = resp.data[i];

if(item.createTime === this.currentEndCreateTime){

this.currentEndCount++;

}

}

}

}

});

hql语句拼接工具类库:https://gitee.com/xuhaohai/jpaDao

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值