web-dev-qa-db-ja.com

ルビーのシングルトンクラスとは正確には何ですか?

シングルトンクラスはRuby自体のクラスですか?すべてのオブジェクトが「クラス」に属する理由ですか?概念はファジーですが、私は信じていますクラスメソッドをまったく定義できる理由と関係があります(class foo; def foo.bar ...)。

Rubyのシングルトンクラスとは何ですか?

74
keyofnight

まず、小さな定義:シングルトンメソッドは、単一のオブジェクトに対してのみ定義されるメソッドです。例:

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のシングルトンメソッドです
  • オブジェクトのシングルトンメソッドは、オブジェクトのクラスのインスタンスメソッドではありません。むしろ、オブジェクトのシングルトンクラスのインスタンスメソッドです。
138
Pistos

Rubyには、特定のオブジェクトに固有のメソッドを定義する方法が用意されており、そのようなメソッドはシングルトンメソッドと呼ばれます。オブジェクトでシングルトンメソッドを宣言すると、Rubyはシングルトンメソッドのみを保持するクラスを自動的に作成します。新しく作成されたクラスはシングルトンクラスと呼ばれます。


    foo = Array.new
    def foo.size
      "Hello World!"
    end
    foo.size  # => "Hello World!"
    foo.class # => Array
    #Create another instance of Array Class and call size method on it
    bar = Array.new
    bar.size  # => 0

singleton_methodsをオブジェクトで呼び出して、オブジェクトのすべてのシングルトンメソッドの名前のリストを取得できます。

    foo.singleton_methods  # => [:size]
    bar.singleton_methods  # => []

この 記事 は、Rubyのシングルトンクラスを理解するのに本当に役立ち、良いコード例があります。

28
Bedasso

@Pistos回答の更新として、バージョン1.9.2からRubyシングルトンクラスを取得するために新しい構文を追加

 singleton_class = ( class << foo; self; end )

次のものに置き換えることができます。

singleton_class = foo.singleton_class

https://apidock.com/Ruby/Object/singleton_class

7
Piotr Galas

最も実用的でアクション重視の方法(IMHO)は、継承チェーン、またはメソッドのルックアップ/解決順序です。この写真は役立つかもしれません

http://www.klankboomklang.com/2007/11/25/modules-part-i-enter-the-include-class/

これはr 1.9であり、組み込みクラスとユーザー定義クラスの対比です。私はまだこれを消化しています。

http://d.hatena.ne.jp/sumim/20080111/p1

また、「シングルトンオブジェクト」という用語の紛らわしい使用法も考えますが、これは異なる概念です。シングルトンオブジェクトは、コンストラクタ/インスタンス化メソッドがオーバーライドされているクラスから取得されるため、そのクラスの1つのみを割り当てることができます。

4
Gene T