Во-первых, небольшое определение: метод singleton - это метод, который определен только для одного объекта. Пример:
irb(main):001:0> class Foo; def method1; puts 1; end; end
=> nil
irb(main):002:0> foo = Foo.new
=> #<Foo:0xb79fa724>
irb(main):003:0> def foo.method2; puts 2; end
=> nil
irb(main):004:0> foo.method1
1
=> nil
irb(main):005:0> foo.method2
2
=> nil
irb(main):006:0> other_foo = Foo.new
=> #<Foo:0xb79f0ef4>
irb(main):007:0> other_foo.method1
1
=> nil
irb(main):008:0> other_foo.method2
NoMethodError: undefined method `method2' for #<Foo:0xb79f0ef4>
from (irb):8
Методы экземпляра - это методы класса (т. Е. Определенные в определении класса). Методы класса являются одноэлементными методами в экземпляре класса Class
- они не определены в определении класса. Вместо этого они определены в одноэлементном классе объекта.
irb(main):009:0> Foo.method_defined? :method1
=> true
irb(main):010:0> Foo.method_defined? :method2
=> false
Вы открываете одноэлементный класс объекта с синтаксисом class << obj
. Здесь мы видим, что в этом одноэлементном классе определены одноэлементные методы:
irb(main):012:0> singleton_class = ( class << foo; self; end )
=> #<Class:#<Foo:0xb79fa724>>
irb(main):013:0> singleton_class.method_defined? :method1
=> true
irb(main):014:0> singleton_class.method_defined? :method2
=> true
irb(main):015:0> other_singleton_class = ( class << other_foo; self; end )
=> #<Class:#<Foo:0xb79f0ef4>>
irb(main):016:0> other_singleton_class.method_defined? :method1
=> true
irb(main):017:0> other_singleton_class.method_defined? :method2
=> false
Таким образом, альтернативным способом добавления одноэлементных методов к объекту было бы определение их с открытым одноэлементным классом объекта:
irb(main):018:0> class << foo; def method3; puts 3; end; end
=> nil
irb(main):019:0> foo.method3
3
=> nil
irb(main):022:0> Foo.method_defined? :method3
=> false
В итоге:
- методы всегда должны принадлежать классу (или: быть методами экземпляра некоторого класса)
- нормальные методы принадлежат классу, в котором они определены (т.е. являются методами экземпляра класса)
- являются просто одноэлементными методами
Class
- одноэлементные методы объекта не являются методами экземпляра класса объекта; скорее они являются методами экземпляра синглтон-класса объекта.