同じクラスを再定義すると,前のクラス定義に追加されていきます. メソッドを再定義した場合には後のものが上書きしますので,前のものは 失われます.
Rubyにはクラス変数に相当するものはありません.コンテナクラス
(Array
,Hash
など)をクラス定数にして代りに使ってください.
class Foo
F = [0]
def foo
F[0] += 1
print F[0], "\n"
end
end
class Foo
@a = 123 # (1)
def foo p @a # (2) ... 123でなくnilになる.
end
end
(1)がクラスのインスタンス変数,(2)が通常のインスタンス変数です.(2)は
クラスFoo
のインスタンスに属するのに対し,(1)はFoo
というクラ
スオブジェクト(Class
のインスタンス)に属します.
インスタンスメソッドからクラスのインスタンス変数に直接アクセスすること はできません.
上のように初期化されていないインスタンス変数とみなされ,nil
になり
ます.
特異メソッドは特定のインスタンスに固有のメソッドです.
こんな感じで使います.
foo = Foo.new
def foo.hello
print "Hello\n"
end
foo.hello
クラスにあるメソッドを追加したいが,わざわざサブクラスを作るほどのこと
でもない,といった場合に有効です.
Javaをやってる人は匿名のインナークラスに似てると思うかもしれませんね.
クラスの特異メソッドをクラスメソッドと呼びます.特異メソッドは オブジェクトの固有のメソッドだと説明したばかりですが,Rubyには, メタクラスという概念があり,すべてのクラスは,同名のメタクラスと いうものを持っていて,これは,Classクラスのインスタンスになって います.ここにクラスメソッドが定義されます.
形式的にはクラス名をレシーバーとして呼べるメソッドということに なります.
Class
のインスタンスであるFoo
の特異メソッドを考えてみま
しょう.
class Foo
def Foo.test
print "this is foo\n"
end
end
呼び出す時はこうです.
Foo.test
何か気付きませんか?
そう,これはいわゆるクラスメソッドですね.
もちろんClass
で定義されているメソッドもクラスメソッドとして使えま
す.
すでに 特異メソッドについては触れました.
簡単におさらいするとRubyではオブジェクト(インスタンス)に対してメソッド を追加することができるわけです.
この考えをもう少し進めるとクラスに対する他の操作をオブジェクトに対して も行えるようにしたくなってきませんか?
C なってこないよ,と言わないで(^^;
これを可能にするのが特異クラスという機構です.
class Foo
def hello
print "hello.\n"
end
end
foo = Foo.new
foo.hello
# -> hello.
class << foo
attr :name, TRUE
def hello
print "hello. I'm ", @name, ".\n"
end
end
foo.name = "Tom"
foo.hello
# -> hello. I'm Tom.
なんかすごいですよね.
ではここで問題.
Q. private_class_method
を使わずにクラスメソッドをprivateにす
るにはどうすればよいでしょう?
ヒント: クラスメソッドはクラスの特異メソッドでしたね.
ちょっとトリッキーですが,こんなふうにできます.
class Foo
...
end
class << Foo
def class_method
print "class method\n"
end
private :class_method
end
Foo.class_method " # -> Error
特異メソッドを定義するには,このように特異クラスで定義する方法と 直接 def obj.method という風に定義してしまう方法があります.
ちょっと性格は違いますが,モジュールでは,モジュール関数にする ことにより,特異メソッド(と同時にprivateメソッド)を定義する ことができます.
モジュールの特異メソッドとして,また同時にprivateメソッドして定義されて いるメソッドをRubyではモジュール関数と呼びます.例えば
Math.sqrt(2)
のように用いることも,
include Math
sqrt(2)
のようにinclude
して用いることもでき,とても便利です.
あるメソッドをモジュール関数にするには,モジュール定義の中で
module_function :method_name
とします.
モジュールはインスタンスを作れません. クラスはincludeできません.
モジュールは,クラス(モジュール)にincludeされることにより,多重継承に 相当するMix-inを実現します.これは直接の継承であるサブクラスとは 異なりますが,includeされたクラスは,モジュールとis_a?の関係を 持ちます.
前者では定数を直接参照することができます.後者ではクラス名をつけて参照 しなければなりません.
load
とrequire
はどう違いますかload
はRubyで書かれたソース(*.rb
)のみロードします.
require
は*.o
ファイルもロードします.さらに一度require
し
たファイルは2度require
してもロードしません.
ロードパスも違います.
include
とextend
はどう違いますかinclude
はmodule
をクラス(モジュール)にインクルードして,
メソッドを関数形式で呼べるようにし,extend
は
module
をオブジェクト(インスタンス)にインクルードして,メソッドを
特異メソッドとして追加します.
self
というのは何ですかself
は,メソッドが適用されるオブジェクトそれ自身を表わします.
関数形式のメソッドは,self
をレシーバーとします.