Что такое mattr_accessor в модуле Rails? - PullRequest
100 голосов
/ 09 октября 2008

Я не смог найти это в документации по Rails, но похоже, что 'mattr_accessor' является следствием Module для 'attr_accessor' (getter & setter ) в обычном Ruby классе .

Например. в классе

class User
  attr_accessor :name

  def set_fullname
    @name = "#{self.first_name} #{self.last_name}"
  end
end

Например. в модуле

module Authentication
  mattr_accessor :current_user

  def login
    @current_user = session[:user_id] || nil
  end
end

Этот вспомогательный метод предоставляется ActiveSupport .

Ответы [ 2 ]

171 голосов
/ 09 октября 2008

Rails расширяет Ruby версиями mattr_accessor (модуль доступа) и cattr_accessor (а также версиями * reader / _writer). Поскольку Ruby attr_accessor генерирует методы получения / установки для экземпляров , cattr/mattr_accessor предоставляют методы получения / установки на уровне class или module . Таким образом:

module Config
  mattr_accessor :hostname
  mattr_accessor :admin_email
end

сокращенно от:

module Config
  def self.hostname
    @hostname
  end
  def self.hostname=(hostname)
    @hostname = hostname
  end
  def self.admin_email
    @admin_email
  end
  def self.admin_email=(admin_email)
    @admin_email = admin_email
  end
end

Обе версии позволяют вам обращаться к переменным уровня модуля следующим образом:

>> Config.hostname = "example.com"
>> Config.admin_email = "admin@example.com"
>> Config.hostname # => "example.com"
>> Config.admin_email # => "admin@example.com"
37 голосов
/ 09 октября 2008

Вот источник для cattr_accessor

А

Вот источник для mattr_accessor

Как видите, они практически идентичны.

Что касается того, почему существуют две разные версии? Иногда вы хотите написать cattr_accessor в модуле, чтобы вы могли использовать его для информации о конфигурации , как Авди упоминает .
Однако cattr_accessor не работает в модуле, поэтому они более или менее скопировали код для работы и для модулей.

Кроме того, иногда вам может понадобиться написать метод класса в модуле, чтобы всякий раз, когда любой класс включал модуль, он получал этот метод класса, а также все методы экземпляра. mattr_accessor также позволяет вам сделать это.

Однако во втором сценарии это поведение довольно странное. Соблюдайте следующий код, особенно обратите внимание на @@mattr_in_module бит

module MyModule
  mattr_accessor :mattr_in_module
end

class MyClass
  include MyModule
  def self.get_mattr; @@mattr_in_module; end # directly access the class variable
end

MyModule.mattr_in_module = 'foo' # set it on the module
=> "foo"

MyClass.get_mattr # get it out of the class
=> "foo"

class SecondClass
  include MyModule
  def self.get_mattr; @@mattr_in_module; end # again directly access the class variable in a different class
end

SecondClass.get_mattr # get it out of the OTHER class
=> "foo"
...