9

目前正在保存中,我正在尝试检查记录是否属于特定的“范围”。这个“范围”实际上只是 .where 调用的一些保存参数。同样在这个“范围”中,我只检查对象的值,而不检查它与数据库中其他对象的关系,所以如果有意义的话,查询数据库总是会过度杀戮。

我只能想出以下解决方案

begin
   result = self.class.where(scope).find(self.id)
rescue
  result = false
end

这样做的问题是,即使我已经有了记录,我也必须查询数据库,而且我不仅要在保存之前运行它,还要在保存之后运行它来检查它的值和保存后的值,因为有如果尚未保存,则无法查询数据库以获取更新版本。

可能有很多这样的检查,所以我想避免必须做两次,也不必多次查询数据库,即使最终我只是通过 id 查找一些东西。

我能想到的唯一其他解决方案是拥有一种方法,该方法可以将 where 调用转换为在传递对象时返回布尔值的 proc。唯一的问题是翻译它必须如何使用正在使用的活动记录适配器,这似乎是一个完整的项目。那么有没有人知道一些方法可以做到这一点,或者有帮助的宝石?

PS我从缓存中获取“范围”,所以我不能将它保存为proc,因为你不能用Rails将procs放入缓存中。

4

2 回答 2

6

首先你可以稍微改进你的第一个解决方案

result = self.class.where(scope).exists?(self.id)

如果您不想检查数据库,为什么不检查您的对象的属性是否具有范围的值?如果您的范围是

class.where(:attr1 => value1, :attr2 => value2, :attr3 => value3)

那么你可以做

result = self.attr1 == value1 and self.attr2 == value2 and self.attr3 == value3
于 2012-07-30T02:58:29.490 回答
4

如果您的范围很简单,您可能希望避免代码重复。我的解决方案允许您调用model.active?以了解实例是否属于范围,并Model.active查找与范围匹配的所有记录。model.active?不涉及任何数据库查询。

考虑将其添加到config/initializers/scope_and_method.rb

require 'active_record/named_scope'

module ActiveRecord::NamedScope::ClassMethods
  def scope_and_method field, *values
    field = field.to_sym
    values.each do |value|
      named_scope value.to_sym, :conditions => {field => value}
      define_method "#{value}?" do
        send(field.to_sym) == value
      end
    end
  end
end

用法:

scope_and_method :state, 'active', 'inactive'

就好像它是:

named_scope :active, :conditions => {:state => 'active'}
named_scope :inactive, :conditions => {:state => 'inactive'}

def active?
  state == 'active'
end

def inactive?
  state == 'inactive'
end

这是 Rails 2.3 的解决方案。这需要对 Rails 3 和 4 进行非常小的调整。( named_scope-> scope) 我会尽快检查。

于 2013-12-18T14:07:26.903 回答