我想知道是否仍有任何方法可以使用.order("RANDOM()")
和will_paginate
,以便在页面加载并对页面进行排序时,所有帖子将在每个页面上保持不变,直到主页重新加载。
以便localhost:3000/posts?page=1
上的所有帖子保持不变,直到再次访问localhost:3000(root_path)
。
问题是它会对帖子进行分页,但它会为每个选定的页面重新排序,因此您经常会在第2页上看到第1页上的帖子。
答案 0 :(得分:1)
执行此操作的一种方法是设置数据库排序的random seed,以便每次都返回相同的随机数序列。您可以将此种子存储在用户的会话中,并仅在需要时重置它。然而,有一个复杂的问题 - 即使设置随机种子每次产生相同的随机数排序,也不能保证你的数据库每次都会以相同的顺序在你的行上执行它,除非你强迫它这样做:
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存储到会话中,当你需要生成下一页时,只需忽略你所做的任何事情已经向用户显示。代码看起来像:
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
此方法使用会话来存储每个页面上显示的ID,因此您可以保证相同的项目集,并且如果用户返回,将显示排序。对于新页面,它将排除已显示的任何项目。您可以随时重置缓存:
session.delete(:pages)
希望有所帮助!您也可以使用Redis或Memcache来存储您的页面数据,但如果您希望按用户随机排序,则会话是一个不错的选择。