分享
  1. 首页
  2. 文章

当查询的数据来自多个数据源,有哪些好的分页策略?

新亮 · · 1115 次点击 · · 开始浏览
这是一个创建于 的文章,其中的信息可能已经有所发展或是发生改变。

概述

在业务系统开发中,尤其是后台管理系统,列表页展示的数据来自多个数据源,列表页需要支持分页,怎么解决?

问题

fa-1-1.jpg

如上图,数据源可能来自不同 DB 数据库,可能来自不同 API 接口,也可能来自 DB 和 API 的组合。

我这也没有太好的解决方案,接到这样的需求,肯定首先和需求方沟通,这样分页是否合理。

无非就两种方案:

  • 数据定期同步,首先将查询的数据汇总到一个地方,然后再进行查询分页。
  • 内存中分页,首先将查询的数据存放到内存,然后再进行查询分页。

如果以某一数据源进行分页,其他字段去其他数据源获取,这样还好处理一些。

如果以多个数据源融合后再分页的话,就数据定期同步 或 内存中分页吧。

数据定期同步方案可以根据实际情况去设计同步频率,至于同步到 ES/MySQL/MongoDB 自己决定即可。

关于内存中分页方案,下面分享两个小方法,供参考。

PHP 方法

$data = [
 0 => ['name' => "姓名1", 'age' => "年龄1"],
 1 => ['name' => "姓名2", 'age' => "年龄2"],
 2 => ['name' => "姓名3", 'age' => "年龄3"],
 3 => ['name' => "姓名4", 'age' => "年龄4"],
 4 => ['name' => "姓名5", 'age' => "年龄5"],
 5 => ['name' => "姓名6", 'age' => "年龄6"],
 6 => ['name' => "姓名7", 'age' => "年龄7"],
 7 => ['name' => "姓名8", 'age' => "年龄8"],
 8 => ['name' => "姓名9", 'age' => "年龄9"],
 9 => ['name' => "姓名10", 'age' => "年龄10"],
];
/**
 * 数组分页
 * @param array $arrayData 数组数据
 * @param int $page 第几页
 * @param int $pageSize 每页展示条数
 * @return array
 */
function arrayToPageData($arrayData = [], $page = 1, $pageSize = 10)
{
 $arrayData = array_values((array)$arrayData);
 $pageData['list'] = array_slice($arrayData, ($page - 1) * $pageSize, $pageSize);
 $pageData['pagination']['total'] = count($arrayData);
 $pageData['pagination']['currentPage'] = $page;
 $pageData['pagination']['prePageCount'] = $pageSize;
 return $pageData;
}
echo json_encode(arrayToPageData($data, 2, 3));

输出:

{
 "list": [
 {
 "name": "姓名4",
 "age": "年龄4"
 },
 {
 "name": "姓名5",
 "age": "年龄5"
 },
 {
 "name": "姓名6",
 "age": "年龄6"
 }
 ],
 "pagination": {
 "total": 10,
 "currentPage": 2,
 "prePageCount": 3
 }
}

Go 方法

package main
import (
 "encoding/json"
 "fmt"
)
type User []struct {
 Name string `json:"name"`
 Age string `json:"age"`
}
type Pagination struct {
 Total int `json:"total"`
 CurrentPage int `json:"currentPage"`
 PrePageCount int `json:"prePageCount"`
}
type ListPageData struct {
 List User `json:"list"`
 Pagination Pagination `json:"pagination"`
}
func main() {
 jsonStr := `[{"name": "姓名1","age": "年龄1"},
 {"name": "姓名2","age": "年龄2"},
 {"name": "姓名3","age": "年龄3"},
 {"name": "姓名4","age": "年龄4"},
 {"name": "姓名5","age": "年龄5"},
 {"name": "姓名6","age": "年龄6"},
 {"name": "姓名7","age": "年龄7"},
 {"name": "姓名8","age": "年龄8"},
 {"name": "姓名9","age": "年龄9"},
 {"name": "姓名10","age": "年龄10"}
 ]`
 var user User
 err := json.Unmarshal([]byte(jsonStr), &user)
 if err != nil {
 fmt.Println(err.Error())
 }
 page := 2
 pageSize := 3
 pageData := ArraySlice(user, page, pageSize)
 listPageData := ListPageData{}
 listPageData.List = pageData
 listPageData.Pagination.Total = len(user)
 listPageData.Pagination.CurrentPage = page
 listPageData.Pagination.PrePageCount = pageSize
 jsonData, _ := JsonEncode(listPageData)
 fmt.Println(jsonData)
}
func JsonEncode(v interface{}) (string, error) {
 bytes, err := json.Marshal(v)
 if err != nil {
 return "", err
 }
 return string(bytes), nil
}
func ArraySlice(u User, page int, pageSize int) User {
 offset := (page - 1) * pageSize
 if offset > int(len(u)) {
 panic("offset: the offset is less than the length of u")
 }
 end := offset + pageSize
 if end < int(len(u)) {
 return u[offset:end]
 }
 return u[offset:]
}

输出:

{
 "list": [
 {
 "name": "姓名4",
 "age": "年龄4"
 },
 {
 "name": "姓名5",
 "age": "年龄5"
 },
 {
 "name": "姓名6",
 "age": "年龄6"
 }
 ],
 "pagination": {
 "total": 10,
 "currentPage": 2,
 "prePageCount": 3
 }
}

小结

如果你有更好的方案,欢迎留言评论 ~

推荐阅读

qr.jpg


有疑问加站长微信联系(非本文作者)

入群交流(和以上内容无关):加入Go大咖交流群,或添加微信:liuxiaoyan-s 备注:入群;或加QQ群:692541889

关注微信
1115 次点击
暂无回复
添加一条新回复 (您需要 后才能回复 没有账号 ?)
  • 请尽量让自己的回复能够对别人有帮助
  • 支持 Markdown 格式, **粗体**、~~删除线~~、`单行代码`
  • 支持 @ 本站用户;支持表情(输入 : 提示),见 Emoji cheat sheet
  • 图片支持拖拽、截图粘贴等方式上传

用户登录

没有账号?注册
(追記) (追記ここまで)

今日阅读排行

    加载中
(追記) (追記ここまで)

一周阅读排行

    加载中

关注我

  • 扫码关注领全套学习资料 关注微信公众号
  • 加入 QQ 群:
    • 192706294(已满)
    • 731990104(已满)
    • 798786647(已满)
    • 729884609(已满)
    • 977810755(已满)
    • 815126783(已满)
    • 812540095(已满)
    • 1006366459(已满)
    • 692541889

  • 关注微信公众号
  • 加入微信群:liuxiaoyan-s,备注入群
  • 也欢迎加入知识星球 Go粉丝们(免费)

给该专栏投稿 写篇新文章

每篇文章有总共有 5 次投稿机会

收入到我管理的专栏 新建专栏