这是我参与更文挑战的第2天,活动详情查看: 更文挑战
首先抛出我的问题?
如何用一个接口实现2个list的翻页加载,需求如下图所示,当list1数据全部加载完毕之后再返回list2的数据。(不限制list1和list2的数据个数,有多少条都要通过翻页加载取出来)
思路
- 关键问题是如何定义翻页,什么场景下开始返回list2的数据
- 翻页加载标配字段pageCount,来定义每页加载数量
- track定义翻页规则,用|分割,|前面是偏移量,|后面标记本次返回哪个list的数据
- 将list1和list2对应的返回场景定义为常量,方便区分
核心代码
const TYPE_LIST1 = 1;
const TYPE_LIST2 = 2;
public static function demoList($userid, $pageCount = 10, $track = '', $select = '*')
{
$offsetTrack = 0;
$typeTrack = 1;
$more = 1;
$list1 = [];
$list2 = [];
if (!empty($track)) {
$decodeTrack = explode("|", $track);
$offsetTrack = intval($decodeTrack[0]);
$typeTrack = intval($decodeTrack[1]);
}
//取list1 或者 list1+list2
if ($typeTrack == self::TYPE_LIST1) {
$list1 = self::getList1($userid, $offsetTrack, $pageCount);
//list1返回的个数比pageCount小 说明数据不足 需要在请求list2的数据
$list1Count = count($list1);
if ($list1Count < $pageCount) {
$typeTrack = self::TYPE_LIST2;
$offsetTrack = $list1Count;
} else {
$track = $offsetTrack + $list1Count . "|" . self::TYPE_LIST1;
$more = 1;
}
}
//只取list2
if ($typeTrack == self::TYPE_LIST2) {
$offset = $offsetTrack;
$list2 = self::getList2($userid, $pageCount, $offset);
$more = count($list2) < $pageCount ? 0 : 1;
$track = $offset + count($list2) . "|" . self::TYPE_LIST2;
}
$ret = [
'track' => $track,
'more' => $more,
'list1' => $list1,
'list2' => $list2,
];
return $ret;
}
复制代码
进阶思路
- 上面代码简化了getList1(),getList()2,目的是直观的解答抛出的问题
- 实际项目中getList()方法内部有缓存数据处理,有考虑数据实时性的问题
进阶问题
- 比如需求变更为list2中的数据有一个状态,如果状态改变需要从list2列表中移动到list1列表中怎么处理?
思考
- 进阶问题考察的关键是状态改变,这个状态改变是实时变化的吗?实时变化的话,只有主动推送状态变化给客户端通知更新。
- 我们便引入了消息推送机制,保证数据结构体的一致性,当list2中的数据状态改变,就推送这条数据到客户端,又客户端添加到list1中
进阶思考
- 这样又引出了一个问题,list2中的数据如何移除,移除之后怎么保证再次翻页查找不会重复取数据,不会丢数据?
© 版权声明
文章版权归作者所有,未经允许请勿转载。
THE END