ruby 基础知识 - Class 与 Module
1. 因為 Ruby 並沒有「屬性」(property/attribute)這樣的設計,要取用實體變數,需要另外定義的方法才行:
class Cat def initialize(name, gender) @name = name @gender = gender end def name @name end def name=(new_name) @name = new_name end end
改写 => 使用 attr_reader
、attr_writer
以及 attr_accessor 定义「讀取」、「設定」以及「讀取 + 設定」的方法,所以原來的有點囉嗦的寫法就可使用
attr_accessor
class Cat attr_accessor :name def initialize(name, gender) @name = name @gender = gender end end
2. 定义类方法
class Cat def self.all # ... end end
或者
class Cat class << self def all # ... end end end
3. 方法的存取控制
注意: initialize
方法,它永遠是 private 的,只會被 new
方法调用。
一般使用
class Cat def eat puts "好吃!" end protected def sleeping puts "zzzzzzzzz..." end private def gossip puts "我跟你說,你不要跟別人說喔!" end end
另一種的方法存取限制是寫在方法定義之後
class Cat def eat puts "好吃!" end def sleeping puts "zzzzzzzzz..." end def gossip puts "我跟你說,你不要跟別人說喔!" end protected :sleeping private :gossip end
使用说明:
1)其實 public
、protected
以及 private
這三個在 Ruby 裡並不是關鍵字,它只是一般的方法而已。
2)private 方法没有调用者 recevier,前面不可以有小数点。
在 Ruby 的 private 方法其實不只類別自己內部可以存取,它的子類別也可以,並沒有像其它程式語言一樣的繼承限制。
比如:
puts "Hello Ruby"
而
self.puts "Hello Ruby" # NoMethodError
3) protected 方法,從外部來看,它跟 private 一樣,不能直接使用。
但在類別內部,它的規定就沒那麼嚴格了,你要指定或不指定 recevier 都可以。
4)private 其实也可以在外部调用
class Cat def say_hello self.gossip end private def gossip puts "我跟你說,你不要跟別人說喔!" end end kitty = Cat.new kitty.gossip # => NoMethodError kitty.send(:gossip) # => 我跟你說,你不要跟別人說喔!
咦?不是說呼叫 private 方法的時候不能有明確的接收者嗎?你仔細看,並沒有違反這個規定喔,這邊我是執行 send
方法,把 gossip
當做參數傳給它而已,所以不算違反規定。
4. 开放类别 Open Class
class Cat def abc # ... end end class Cat def xyz # ... end end kitty = Cat.new kitty.abc # => 會發生什麼事? kitty.xyz # => 會發生什麼事?
1)两个相同的类会进行融合。
上面的两个类会变成这样
class Cat def abc # ... end def xyz # ... end end
除此之外,还可以对内建的类别进行融合
class String def say_hello "hi, I am #{self}" end end puts "eddie".say_hello # => hi, I am eddie puts "kitty".say_hello # => hi, I am kitty
2)Open Class 覆盖原来的方法。
class Integer def +(n) 1000 end end puts 1 + 2 #=> 得到 1000 puts 3 + 4 #=> 得到 1000
改写
class Integer alias :original_plus :+ def +(n) puts "hey hey hey" original_plus(n) end end puts 1 + 2 puts 3 + 4
这里使用了 Ruby 內建的 alias
方法把原本的 +
方法加個別名 original_plus
,然後再新定義的 +
方法裡,再呼叫它原本的算法。執行之後就會發現計算結果跟原本的 +
是一樣的,但會偷偷多印了 hey hey hey
字樣在畫面上。
5. 模组
定义一个模组
module Flyable def fly puts "I can fly!" end end
引入模组
class Cat include Flyable end kitty = Cat.new kitty.fly # => I can fly!
Ruby 中 Include, Extend, Load, Require 的使用区别
「類別」的上層類別就是「模組」,開 IRB 實驗一下:
$ irb >> Class.superclass => Module
区别
$ irb >> Class.instance_methods - Module.instance_methods => [:new, :allocate, :superclass]
可以發現身為「後代」的 Class
類別,比 Module
類別多了 3 個方法,就是因為 Module
類別少了這 3 個方法,所以它跟 Class
最大的差別,就是:
- 模組沒辦法
new
一個新的實體出來。 - 模組沒辦法繼承別的模組。
除此之外,模組跟類別在本質上沒什麼太大的差別。
末尾!