myBatis无限滚动的偏移位置设置问题

Posted

技术标签:

【中文标题】myBatis无限滚动的偏移位置设置问题【英文标题】:myBatis offset position setting issue for infinite scroll 【发布时间】:2019-01-14 01:48:58 【问题描述】:

我正在构建一个类似 Quora 的应用程序。后端使用spring boot、mybatis连接一个mysql数据库。当用户打开网站时,后端会返回前 10 个问题。如果用户点击“获取更多”按钮,后端应该返回接下来的 10 个问题。

mybatis的代码是

<mapper namespace="com.quora.dao.QuestionDAO">
    <sql id="table">question</sql>
    <sql id="selectFields">id, title, content, comment_count,created_date,user_id
    </sql>
    <select id="selectLatestQuestions" resultType="com.quora.model.Question">
        SELECT
        <include refid="selectFields"/>
        FROM
        <include refid="table"/>

        <if test="userId != 0">
            WHERE user_id = #userId
        </if>
        ORDER BY id DESC
        LIMIT #offset,#limit
    </select>
</mapper>

目前我的逻辑是第一次#offset为0,第二次#offset为10。但是当表频繁更新时我发现这个逻辑不正确。如果表已插入新行,用户可能会得到重复数据。如何根据前端显示的最后一个问题 ID 设置 #offset?比如前端最后一个question id是10,那么#offset应该是question id 10的行号。

谁能给我一些建议?

谢谢, 彼得

【问题讨论】:

【参考方案1】:

一般的想法是根本不使用OFFSET,而是进行过滤。如果您可以定义消息的顺序,以便在插入新消息时它不会更改(例如,您逐步生成 id 并按id ASC 对消息进行排序),那么这很容易:

SELECT id, some_other_field, yet_another_field
FROM question
<if test="last_seen_question_id != null">
    WHERE id > #last_seen_question_id
</if>
ORDER BY id ASC
LIMIT #limit

然后客户端应该使用最后看到的问题 id 并在它想要获取下一页时传递它。

从您的查询 (ORDER BY id DESC) 看来,您希望在顶部看到最新的问题。这有点问题,因为新插入的问题往往会到达顶部。

如果您可以先在下一页上获得新问题,然后是旧问题,您可以这样做:

<!-- This condition is needed to avoid duplication when the first page is fetched
     and we haven't seen any question yet.
     In this case we just get up to limit of the last questions.
-->
<if test="newest_seen_question_id != null">
SELECT * FROM (
  -- select all questions that were created _after_
  -- the last seen max question id
  -- client should maintain this value as in get the 
  -- largest question id for every page and compare it
  -- with the current known max id. And client should
  -- do it for the whole duration of the paging
  -- and pass it to the follow up queries
  SELECT id, some_other_field, yet_another_field
  FROM question
  WHERE id > #newest_seen_question_id
  -- note that here we do sorting in the reverse order
  -- basically expanding the set of records that was returned
  -- in the direction of the future
  ORDER BY id ASC
  LIMIT #limit
  UNION ALL
</if>
  -- select all questions that were created _before_
  -- the last seen min question id.
  -- client should maintain this value for the whole
  -- duration of the paging
  -- and pass it to the follow up queries      
  SELECT id, some_other_field, yet_another_field
  FROM question
  <if test="oldest_seen_question_id != null">
    WHERE id < #oldest_seen_question_id
  </if>
  ORDER BY id DESC
  LIMIT #limit
<if test="newest_seen_question_id != null">
) AS q
ORDER BY id DESC
LIMIT #limit
</if>

另一个好处是,从性能的角度来看,这种不使用OFFSET 的分页方法是much better。

【讨论】:

以上是关于myBatis无限滚动的偏移位置设置问题的主要内容,如果未能解决你的问题,请参考以下文章

iOS: 无限循环轮播图简单封装

在无限滚动中延迟加载时锁定滚动位置(向上滚动)

添加新数据时,Tableview 会滚动到顶部。无限滚动

画布无限滚动映射对象位置

无限轮播Banner的实现原理

无限滚动返回到相同的位置