我应该使用类方法还是实例方法,为啥?

Posted

技术标签:

【中文标题】我应该使用类方法还是实例方法,为啥?【英文标题】:Should I use class method or instance method, and why?我应该使用类方法还是实例方法,为什么? 【发布时间】:2012-07-24 05:32:12 【问题描述】:

在我的 Rails 应用程序中,创建业务时,我有一个包含以下字段的表单:

   <%= check_box_tag(:default_company) %> 
   <%= label_tag(:default_company, "Set Company as Default") %>

基本上当我创建业务时,如果他们选中此框,我需要它运行类似于以下代码的内容:

def set_default_company(company, user)
  exists = DefaultCompany.find(user.id)
  if exists
    exists.update_attributes(company: company)
  else  
    DefaultCompany.create(company: company, user: user)
  end
end

在学习时,我通常会在我的控制器中做这些事情,但我正在尝试遵循最佳实践并使用胖模型、瘦控制器,所以我想使用这样的逻辑:

def create
  @company = Company.new(params[:company])

  if @company.save
    if params[:default_company]
      Company.set_default_company(@company.id, current_user.id,)
    end
    flash[:notice] = "Company was successfully created."
    redirect_to @company
  else
    redirect_to new_company_path
  end
end

这是我对使用类方法还是实例方法调用set_default_company 感到困惑的地方。他们似乎都可以工作,但我看不出其中一个有什么好处。

除了向我提供有关使用哪种方法的任何信息外,如果有人可以向我展示将其作为类方法与实例方法编写的简要实现,它可能会让我更好地理解原因。

我会这样写:

def self.set_default_company(company, user)
  # Logic here
end

def set_default_company(company, user)
  # Logic here
end

以这种方式写它们我也看不出有什么好处。

【问题讨论】:

【参考方案1】:

顾名思义,模型上的实例方法应该用于与用户的特定实例(调用该方法的实例)相关的逻辑/操作。因此您可以考虑为用户作为User 上的实例方法。类方法适用于不对模型的单个实例进行操作的事物,或者适用于您没有可用实例的情况。例如您可能有一个类方法来整理您的数据库,例如 User.purge_expired_users,它不适用于单个用户对象。

例如

class User
  def set_default_company(company)
    exists = DefaultCompany.find(self.id)
    if exists
      exists.update_attributes(company: company)
    else  
      DefaultCompany.create(company: company, user: self)
    end
  end
end

那么您的控制器方法将如下所示:

def create
  @company = Company.new(params[:company])

  if @company.save
    if params[:default_company]
      current_user.set_default_company @company
    end
    flash[:notice] = "Company was successfully created."
    redirect_to @company
  else
    redirect_to new_company_path
  end
end

或者,您可以从另一个角度考虑关系,并在Company 上放置一个实例方法,例如company.set_as_default_for(user).

【讨论】:

对类与实例的很好描述,实际上真的为我清除了它。你是对的,我将把它转移到我的用户模型而不是公司,这最有意义并且读起来更好。我有最后一个问题。我不应该将默认公司存储在名为 Default Company 的表中,该表是 user_id 和 company_id 的连接表,我应该在用户模型中只包含一个“default_company”字段吗?我选择加入表的原因是因为并非所有用户都会有公司,事实上大多数人不会有公司,所以我会有很多空白字段。【参考方案2】:

我实际上会让set_default_company 成为User 上的实例方法。一个User 有一个默认的Company;为什么Company 需要它默认的用户?

class User
    def set_default_company(company)
        exists = DefaultCompany.find(id)
        if exists
            exists.update_attributes(company: company)
        else  
            DefaultCompany.create(company: company, user: self)
         end
     end
 end

【讨论】:

你是对的。我本来打算这样做,但由于某种原因,我的想法让我远离了那个。【参考方案3】:

在我看来,我总是创建一个class method,如果所讨论的方法代表的信息/行为在所有实例化的对象中非常通用,不同于instance methods,我认为它更像是一个有问题的实例化对象的特定操作。

但这是我的观点。

【讨论】:

【参考方案4】:

一些事情:你有一个 DefaultCompany 的单独表吗?这似乎应该是公司表上的布尔标志。

接下来,公司和用户之间是否存在关联?如果是这样,似乎最好的方法是

在用户模型中

def set_default_company(company)
  self.companies.each do |c|
    c.update_attributes(:default => false)
  end
  company.update_attributes(:default => true)
end

或者在公司模型中

def set_as_default
  update_attributes(:default_company => true)
end

【讨论】:

我确实为 DefaultCompany 提供了一个单独的表,使我远离布尔值的想法是用户通过角色拥有_many 公司,因此用户可能拥有许多公司或没有公司。所以布尔字段在这种情况下实际上不起作用,我不认为 考虑公司有 5 个用户的情况,如果我将公司布尔字段设置为 true,它将为所有用户设置它,对吗? 忘了说这是一个公司通过角色有很多用户,用户也通过角色有很多公司 如果每个用户只能有一个默认公司,那么默认应该是角色表上的布尔值 嗯是的,这是有道理的,理论上用户可以拥有“所有者”和“经理”的角色,所以在我看来,我必须同时跟踪两者?或者我想我会检查用户 a 属于公司 x 的所有角色并将所有这些角色设置为 true,这对您来说似乎比加入表更好?

以上是关于我应该使用类方法还是实例方法,为啥?的主要内容,如果未能解决你的问题,请参考以下文章

内部类方法应该返回值还是只修改实例变量?

为啥在 python 中使用类方法而不是实例方法

python 为啥要使用静态方法

Netty 4 标头:我应该使用静态方法还是实例方法?

在java中,为啥类实例也能访问静态域?

一旦用户登录,我应该使用类实例(序列化)还是 Singleton 来存储 userInfo