【问题标题】:stale association data on ActiveRecord model instance?ActiveRecord 模型实例上的陈旧关联数据?
【发布时间】:2014-05-15 22:27:48
【问题描述】:

我使用的是 Rails 2,我在此应用中的 Project 模型和 Schedule 模型之间存在一对多关联。

当各种事物的属性发生变化时,我有一个观察者检查,作为响应,它会迭代一个哈希数组来填充新的 Schedule 实例。

每个Schedule 实例都应该有一个属性disp_order,当在列表中显示它们时,它最终会告诉前端在哪里显示它。 disp_order 在将计划添加到项目时填充,因此它等于当前最高 disp_order 的 1。

问题在于观察者的迭代。当我遍历填充有Schedule 数据的哈希数组时,每次迭代都应将disp_order 计算为比前一个高一个,以考虑它刚刚添加的Schedule。然而,在实践中,它不会——除非我在迭代过程中使用project = Project.find(project.id) 刷新Project 对象,否则它似乎总是计算disp_orders 的相同最大值,并且没有没有准确的Schedule 实例列表。

有没有更好的方法来做到这一点? (注意:我的意思是有更好的方法,这样我就不必告诉项目重新找到自己。有很多地方我正在积极清理下面的其余代码,但为此问题我真的只对这一行和影响它的事情感兴趣。)

项目.rb

class Project < ActiveRecord::Base
  # ...
  has_many :schedules, :dependent => :destroy
  # ...
  belongs_to :data, :polymorphic => true, :dependent => :destroy
    accepts_nested_attributes_for :data
  # ...

schedule.rb

class Schedule < ActiveRecord::Base
  belongs_to :project, :include => [:data]
  # ...

project_event_observer.rb

class ProjectEventObserver < ActiveRecord::Observer
  # ...          
  def perform_actions(project, actions)
    # ...
    actions.each { |action|
      action.each { |a_type, action_list|
        action_list.each { |action_data|
          self.send(action_type.to_s.singularize, action_data, project)
          project = Project.find(project.id) #My question is about this line.
        }
      }
    }
  # ...

上述迭代的示例操作

[    
  {:add_tasks => [
    {:name => "Do a thing", :section => "General"},
    {:name => "Do another thing", :section => "General"},
   ]
  },
  # More like this one.
]

接收其余动作数据的观察者方法

def add_task(hash, project)
  # ...
  sched = Schedule.new
  hash.each { |key, val|
    # ...
    sched.write_attribute(key, val)
    # ...
  }
  sched.add_to(project)
end

schedule.rb

def add_to(proj)
  schedules = proj.schedules
  return if schedules.map(&:name).include?(self.name)
  section_tasks = schedules.select{|sched| sched.section == self.section}
  if section_tasks.empty?
    self.disp_order = 1
  else
    self.disp_order = section_tasks.map(&:disp_order).max + 1 #always display after previously existing schedules
  end
  self.project = proj
  self.save
end

【问题讨论】:

  • 展示您的问题的控制台示例会很有帮助。

标签: ruby-on-rails ruby activerecord ruby-on-rails-2


【解决方案1】:

您遇到此问题是因为您正在处理内存中缓存的对象。

看看association documentation

您可以在 Schedule 模型的 add_to 函数中传递 true 参数来重新加载查询。

根据文档,其他选项将包括 inverse_of 但您无法使用,因为您使用的是多态关联。

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2014-11-22
    • 1970-01-01
    • 2012-07-06
    • 1970-01-01
    相关资源
    最近更新 更多