【问题标题】:same code in different actions of the controller in rails -- how to make DRY?rails中控制器的不同操作中的相同代码-如何使DRY?
【发布时间】:2010-09-28 20:56:39
【问题描述】:

我有一个具有两个不同操作的控制器,但都需要相同的代码,这有点长,我怎样才能让他们访问相同的行为但保持干燥?

   @list = Contact.find :all,
      :select => "companies.name AS co_name, 
                  companies.id AS comp_id, 
                  COUNT(contact_emails.id) AS email_count, 
                  COUNT(contact_calls.id) AS call_count, 
                  COUNT(contact_letters.id) AS letter_count, 
                  COUNT(contact_postalcards.id) AS postalcard_count",

      :conditions => ['contact_emails.date_sent < ? and contact_emails.date_sent > ?', 
                      report_end_date, report_start_date],

      :joins => [
        "LEFT JOIN companies ON companies.id = contacts.company_id",
        "LEFT JOIN contact_emails ON contact_emails.contact_id = contacts.id",
        "LEFT JOIN contact_letters ON contact_letters.contact_id = contacts.id",
        "LEFT JOIN contact_postalcards ON contact_postalcards.contact_id = contacts.id",
        "LEFT JOIN contact_calls ON contact_calls.contact_id = contacts.id"
      ],
      #:group => "companies.id"
       :group => "companies.name"
    puts @list[0].attributes.inspect

【问题讨论】:

  • 这不是在回答您的问题,但我 90% 确定所有这些“连接”都可以替换为“包含”。像这样::include => [:companies, :contact_emails, :contact_letters, :contact_postalcards, :contact_calls]。

标签: ruby-on-rails controller dry


【解决方案1】:

您应该将此代码移至模型:

# Contatct model

def self.get_list(report_start_date, report_end_date)
  self.find :all,
    :select => "companies.name AS co_name, 
              companies.id AS comp_id, 
              COUNT(contact_emails.id) AS email_count, 
              COUNT(contact_calls.id) AS call_count, 
              COUNT(contact_letters.id) AS letter_count, 
              COUNT(contact_postalcards.id) AS postalcard_count",

    :conditions => ['contact_emails.date_sent < ? and contact_emails.date_sent > ?', 
                  report_end_date, report_start_date],

    :joins => [
      "LEFT JOIN companies ON companies.id = contacts.company_id",
      "LEFT JOIN contact_emails ON contact_emails.contact_id = contacts.id",
      "LEFT JOIN contact_letters ON contact_letters.contact_id = contacts.id",
      "LEFT JOIN contact_postalcards ON contact_postalcards.contact_id = contacts.id",
      "LEFT JOIN contact_calls ON contact_calls.contact_id = contacts.id"
    ],
    #:group => "companies.id"
    :group => "companies.name"
end

然后你可以在控制器中使用它:

@list = Contact.get_list(report_start_date, report_end_date)

也许您也可以将其拆分为更小的部分并使用范围和定义的关联,而不是自己编写所有内容。

【讨论】:

  • 绝对是。这段代码在尖叫,“让我离开控制器!”将其推送到模型会清理您的代码,并允许您从 rake 任务中访问该方法,或者如果您想提取该列表以获得一次性报告,则可以更轻松地从控制台访问该方法。
  • 这段代码看起来是否正确,我很想清理它,但我找不到方法......
  • Join 通常很难看,重要的是它是 DRY:你只需要在模型中定义一次,但你可以多次调用它是各种控制器,正如 klew 指出的那样。
【解决方案2】:

我会添加一个函数来生成计数并加入 sql:

class Contact < ActiveRecord::Base
  def  self.get_list(report_start_date,    report_end_date)
    all(:select      => "companies.name AS co_name,
                        companies.id AS comp_id,
                        #{table_count_col(
                          :contact_emails, 
                          :contact_calls, 
                          :contact_letters, 
                          :contact_postalcards
                         )}",
        :conditions => ['contact_emails.date_sent < ?  AND
                         contact_emails.date_sent > ?',
                         report_end_date, report_start_date],
        :joins      => join_table(
                    :companies, 
                    :contact_emails, 
                    :contact_letters, 
                    :contact_postalcards, 
                    :contact_calls
                    ),
       )
  end
end

其中table_count_coltable_joinContact 类中的静态方法:

  def self.table_count_col(*args)    
    args.collect do |table| 
      count_col = "#{table.to_s.gsub(/^contact_/,  '').singularize}_count"
      "COUNT(#{table}.id) AS #{count_col}"    
    end.join(",")
  end

  def self.table_join(*args)    
    args.collect do |table|
     "LEFT JOIN #{table} ON #{table}.id = contacts.company_id"
    end.join(",")
  end       

【讨论】:

  • table_count_col 和 table_join 一样是 Contacts 上的一种方法?
  • 是的,它们是 Contact 类的静态方法。
  • 我喜欢 +1 以帮助简化 - 可能需要分阶段实施
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多