如何保持ActiveRecord关联DRY?

时间:2008-10-16 21:45:57

标签: mysql activerecord dry

我在尝试将AR查找器DRY保留在我的应用程序中时遇到问题。我创建了一个博客应用程序,当用户第一次查看博客时,它会获取博客的所有相关页面,帖子,链接,标签和类别。博客控制器的示例显示操作如下所示:

def show
    #find blog by user name
    @user= User.find_by_login(params[:id])
    @blog= @user.blog
    @posts = Post.status("publish",@user).find(:all, :order => "created_at DESC")
    @tags = @user.tags
    @pages = Page.status("publish",@user).find(:all, :order => "created_at DESC")
    @links = @user.links.public_link.find(:all, :order => 'created_at DESC')
    @archives = @posts.group_by(&:month)
    @categories = @user.categories.group_by(&:name)
    session[:found_user]=@user.login
    render :layout=>false
  end

正如您所看到的那样,它不是很干,因为还有其他动作会在控制器中进一步调用相同的实例变量,例如@tags等。

我怎么能让这更干?我尝试将其移入Blog模型,但我仍然需要在控制器中调用各种实例变量,如@tags等。

有没有办法在首次调用博客时存储所有这些变量,并在控制器和操作之间重复使用它们?

感谢您的任何建议。我正在使用Rails 2.1

1 个答案:

答案 0 :(得分:3)

我在某个博客上阅读过简单替换过滤器(或在控制器方法中加载各种数据)之前用辅助方法。像这样:

class BlogsController < ApplicationController
  def show
    session[:found_user]=@user.login
    render :layout=>false
  end

  helper_method :user, :blog, :posts, :tags, :pages, :links, :archives, :categories

  protected
  def user
    @user ||= User.find_by_login(params[:id])
  end

  def blog
    @blog ||= user.blog
  end

  def posts
    @posts ||= Post.status("publish", user).find(:all, :order => "created_at DESC")
  end

  def tags
    @tags ||= user.tags
  end

  def pages
    @pages ||= Page.status("publish", user).find(:all, :order => "created_at DESC")
  end

  def links
    @links ||= user.links.public_link.find(:all, :order => 'created_at DESC')
  end

  def archives
    @archives ||= posts.group_by(&:month)
  end

  def categories
    @categories ||= user.categories.group_by(&:name)
  end
end

## app/views/blogs/show.html.erb
<p>Name: <%=h user.name %></p>
<h3><%= posts.length %> Posts</h3>
<% posts.each do |post| %>
  ...
<% end %>
<ul>
  <% categories.each do |category| %>
    <li><%=h category %></li>
  <% end %>
</ul>

在视图中查看如何使用任何简单的数据库调用。此解决方案的一个优点是,未调用的辅助方法不会花费时间。

如有必要,将辅助方法抽象为模块,并将模块包含在ApplicationController中。