2

这里有很多与使用 CouchDB 进行分页相关的问题,但没有一个完全符合我的疑惑。

基本上,我有一个按票数排名的结果集,我想按降序翻页。

这是map供参考的。

function(doc) {
  emit(doc.votes);
}

现在,问题。我发现startkey_docid它自己不起作用。您必须将它与startkey. 问题是,对于查询,我不使用startkey参数(我不想限制结果,只是获取最多-> 最少)。我在想我可以只使用 startkey={{doc.votes}}&startkey_docid={{doc._id}} 代替,但是当有人单击“下一页”链接时,文档的投票数可能已经改变。

解决这个问题的方法似乎很明显:只需设置startkey=99999999它返回数据库中的所有文档,我就可以使用startkey_docid我们上次离开的那个开始。奇怪的是,当我这样做时,startkey_docid停止工作并允许再次返回所有结果。显然startkey需要完全等于文档_id中使用的键startkey_docid

我要问的是,是否有人知道在您想要使用它startkey_docid时实际可能已经改变的页面的解决方法?startkey我的应用程序是否应该只查找文档_id并立即使用doc.votes希望它在请求之间的几毫秒内没有改变的值?即使这样似乎也不是很可靠。

编辑:最终切换到 Mongo 以提高速度,所以这个问题结果有点没有实际意义。

4

3 回答 3

1

我从来没有做过这样的事情,但我想我知道该怎么做。您可以做的是对评级进行快照并在每个页面中引用它。您可能希望您的视图不占用太多空间,因此您不应该映射文档的单独副本,其中在拍摄快照后投票未更改。因此,您可以执行以下操作:

  1. 在您的文档中添加一些带有时间戳的评级历史记录。
  2. 像这样映射评级和历史。
  3. 在您的应用程序中获取当前时间:start_time = Date.now()并查询所有页面。
  4. 清理比最旧的活动会话更早的历史记录。

问题是,如果您发出[votes, date]并尝试分页,您将永远不知道必须获取多少文档才能获得每页所需的数量。总有一些旧版本你必须跳过,你将不得不从 DB 获取下一个版本。这就是为什么你可以考虑发出: [date, votes],总是读取视图两次——对于start_time当前时间,然后合并和排序结果(就像在合并排序中一样)。

广告 1:

{ ...,
  votes: 12,
  history: [
    {date: 1357390271342, votes: 10},
    {date: 1357390294682, votes: 11}
  ]
}

广告 2:

function (doc) {
  emit([{}, doc.votes], null);
  doc.history && doc.history.forEach(function(h) {
    emit([h.date, h.votes], null);
  });
}

广告 3:

?startkey=[start_time, votes]&limit=items_per_page_plus1
?startkey=[{}, votes]&limit=items_per_page_plus1

合并列表,votes在您的应用程序中排序(在列表功能中)。如果您在使用时遇到问题,start_docid那么您可以使用 ID 显式发出[date, votes, id]和查询。即使这个特定的文档更改了votes它,它仍然会在历史记录中可用。

Ad.4:如果您发出[date, votes],那么您可以获得过时的历史宽度:?startkey=[0]&endkey=[oldest_active_session_time]&inclusive_end=false并使用更新处理程序更新它们:

function(doc, req) {
  if (!doc || !doc.history) return [null, 'Error'];
  var history = new Array();
  var oldest = +(req.query.date);
  doc.history.forEach(function(h) {
    if (h.date >= oldest)
      history.push(h);
  });
  doc.history = history;
  return [doc, 'OK'];
}

注意:我没有测试过,所以如果不修改,预计不会运行:)

据我所知,CouchDB 使用 b-tree 阴影进行更新,原则上应该可以访问旧版本的视图。我不喜欢 CouchDB 设计,所以这只是一个猜测,似乎没有任何(记录在案的)API。

于 2013-01-05T13:53:09.053 回答
0

我有点惊讶这个问题没有得到解答,因为 CouchDB Futon 的功能基本上是在您对 map 函数的结果进行分页时完成的。我打开 firebug 以查看在分页时 javascript 控制台中发生了什么,并看到对于每组分页结果,它都将 startkey 与 startkey_docid 一起传递。因此,尽管问题是如何在不包含 startkey 的情况下进行分页,但 CouchDB 指定 startkey 是必需的,并演示了它是如何工作的。endkey没有指定,所以如果指定的startkey只有一个结果,那么下一组分页结果也会包含排序结果中与startkey不匹配的下一个key。

所以澄清一下,这个问题的答案是,当您分页并跟踪 startkey_docid 时,您还需要捕获同一文档的 startkey,它将成为下一组结果的开始。当您调用分页结果时,请同时使用捕获的 startkey 和 startkey_docid 作为 couchdb 的要求。关闭 endkey 以便结果将继续到排序结果的下一个键。

希望能够在不指定键的情况下进行分页的用例场景有点奇怪。因此,假设下一个分页结果的起始 docid 确实将其关键值从 9 大幅更改为 3。我们还假设地图结果中只存在一个 docid 实例,即使它可能出现多次(我相信这就是需要指定 startkey 的原因)。当用户单击下一个按钮时,用户的分页结果现在将从查看 rank 9 移动到 rank 3。但是如果除了 startkey_docid 之外还包括 startkey,则分页结果将从头开始排名 9 的结果比潜在地跳过大量结果更合乎逻辑。

于 2014-03-18T16:29:42.967 回答
0

我现在想不出任何简单的解决方案,但有一些选择:

  • 不经常将您的排序列表复制到小型专用数据库,这样它就会比 stale=ok 更陈旧
  • 修改您的架构,以便您能够按一些更稳定的数据进行排序。查看 CouchDb 指南中的银行/分类帐示例:http: //guide.couchdb.org/draft/recipes.html#banking。例如,尝试记录每一次投票并每小时减少一次。作为奖励,您将获得历史/趋势:)
于 2013-01-05T14:05:14.740 回答