gpt4 book ai didi

ruby-on-rails - .order ("RANDOM()") 与 will_paginate gem

转载 作者:数据小太阳 更新时间:2023-10-29 08:08:04 30 4
gpt4 key购买 nike

我想知道是否有任何方法仍然可以将 .order("RANDOM()")will_paginate 一起使用,这样当页面加载并排序时页面,所有帖子将在每个页面上保持不变,直到重新加载主页。

所以要让 localhost:3000/posts?page=1 上的所有帖子保持不变,直到再次访问 localhost:3000(root_path)

问题是它会对帖子进行分页,但它目前会为所选的每个页面重新排序,因此您经常会在第 1 页和第 2 页看到帖子。

最佳答案

一种方法是设置 random seed您的数据库正在排序,这样它每次都会返回相同的随机数序列。您可以将此种子存储在用户的 session 中,并仅在需要时重置它。然而,有一个复杂的问题——即使设置随机种子每次都会产生相同的随机数顺序,也不能保证你的数据库每次都会以相同的顺序在你的行上执行它,除非你强制它这样做:

SELECT items.*
FROM (SELECT setseed(0.2)) t
, (SELECT name, rank() OVER (ORDER BY name DESC)
FROM foos ORDER BY name DESC) items
JOIN generate_series(1, (SELECT COUNT(*) FROM foos))
ON items.rank = generate_series
ORDER BY RANDOM()
LIMIT 10;

如您所知,这非常复杂,它会迫使您的数据库将整个表具体化到内存中。它适用于较小的数据集,但如果你有一个大数据集,那就不可能了!

相反,我建议您使用更像 tadman suggested above 的解决方案:生成一页结果,将 id 存储到 session 中,当您需要生成下一页时,只需忽略您已经向用户显示的任何内容。代码如下所示:

class ThingsController < ApplicationController
def index
@page = params[:page].to_i
session[:pages] ||= {}

if ids = session[:pages][@page]
# Grab the items we already showed, and ensure they show up in the same order.
@things = Things.where(id: ids).sort_by { |thing| ids.index(thing.id) }
else
# Generate a new page of things, filtering anything we've already shown.
@things = Things.where(["id NOT IN (?)", shown_thing_ids])
.order("RANDOM()")
.limit(30) # your page size
# Save the IDs into our session so the above case will work.
session[:pages][@page] = @things.map(&:id)
end
end

private
def shown_thing_ids
session[:pages].values.flatten
end
end

此方法使用 session 来存储每个页面上显示的 ID,因此您可以保证在用户返回时显示相同的项目集和顺序。对于新页面,它将排除任何已显示的项目。您可以随时重置缓存:

session.delete(:pages)

希望对您有所帮助!您也可以使用 Redis 或 Memcache 来存储您的页面数据,但如果您希望每个用户的顺序是随机的, session 是一个不错的选择。

关于ruby-on-rails - .order ("RANDOM()") 与 will_paginate gem,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31729711/

30 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com