筆記一:
代碼中包含變量,類和方法,統稱為語言構建(language construct)。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
# test.rb class Greeting def initialize(text) @text = text end def welcome @text end end my_obj = Greeting.new( "hello" ) puts my_obj. class puts my_obj. class .instance_methods(false) #false means not inherited puts my_obj.instance_variables result = > Greeting welcome @text |
總結:
實例方法繼承于類,實例變量存在于對象本身。
類和對象都是ruby中的第一類值。
應用示例:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
|
mongo API for ruby => Mongo::MongoClient # testmongo.rb require 'mongo' require 'pp' include Mongo # the members of replcation-set # test mongodb server version 2.6.0 host = "192.168.11.51" # The port of members # If the port is 27017 by default then otherport don't need to assignment otherport = "" port = otherport.length != 0 ? otherport : MongoClient:: DEFAULT_PORT opts = { :pool_size => 5 , :pool_timeout => 10 } # Create a new connection client = MongoClient. new (host, port, opts) # puts client.class puts client. class .constants puts client.instance_variables puts client. class .instance_methods( false ) |
分別輸出
1
|
Constant, Instance Attribute, Instance Method |
筆記二:動態調用
當你調用一個方法時,實際是給一個對象發送了一條消息。
1
2
3
4
5
6
7
8
9
10
|
class MyClass def my_method(args) args * 10 end end obj = MyClass. new puts obj.my_method( 5 ) puts "**" puts obj.send( :my_method , 6 ) |
結果:
1
2
3
|
50 ** 60 |
可以使用object#send()取代點標記符來調用MyClass#my_method()方法:
1
|
obj.send( :my_method , 6 ) |
send()方法第一個參數是要發送給對象的消息,可以是符號(:symbol)或字符串,其他參數會直接傳遞給調用的方法。
可以動態的決定調用哪個方法的技術,成為Dynamic Dispatch。
筆記三:符號和字符串的區別
1. 符號不可變,可以修改字符串中的字符。
2. 針對符號的操作更快些。
3. 通常符號用來表示事物的名字。
例如:
1
2
3
4
5
|
puts 1 .send(:+, 4 ) => 5 String #to_sym(),String#intern() => string to symbol String #to_s(),String#id2name() => symbol to string "caoqing" .to_sym() => :caoqing :caoqing .to_s() => "caoqing" |
動態派發中使用模式派發(pattern dispatch)的方法。
1
2
3
|
puts obj. class .instance_methods( true ).delete_if{ |method_name| method_name !~ /^my/} result => my_method |
筆記四:動態定義
使用Module#define_method()方法定義一個方法。
1
2
3
4
5
6
7
|
class MyClass define_method :my_method do |args| args * 3 end end obj = MyClass. new puts obj.my_method( 10 ) |
結果:30
單件方法允許給單個對象增加一個方法。singleton methods
1
2
3
4
5
6
7
8
9
|
# test.rb str = "My name is caoqing." def str.title? self .upcase == self end puts str.title? puts str.methods.grep(/^title?/) puts str.singleton_methods |
結果:
1
2
3
|
false title? title? |
筆記五:
類方法的本質,類是對象,類名是常量。在類上調用方法和對象調用方法一樣:
1
2
|
obj.my_method Cla.class_method |
Duck Typing:對象能不能響應方法,可以是普通方法或者單件方法。
類方法的實質就是他們是類的一個單件方法。
1
2
3
|
def obj.method # method body end |
obj可以是對象引用,常量類名或self。
類宏(Class Macro)
Ruby對象沒有屬性,可以使用擬態方法定義屬性。
Module#attr_*()方法中的一員來定義訪問器。類宏不是關鍵字而是方法。
Eigenclass
單件方法按照常規的方法查找在祖先鏈無法找到保存的地方,obj是對象不能保存,也不能存在于class內,否則所有的實例都可以共享這個方法。
對象擁有一個特有的隱藏類,稱為該對象的eigenclass。
進入eigenclass作用域:
1
2
3
|
class << obj code end |
如果想獲取eigenclass的引用,則可以在離開該作用域時返回self:
附錄:
類變量,實例變量,類方法,實例方法區別
@@ :var類變量
@ :實例變量
self(?clas,::).method :類方法
method :實例方法
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
|
# test.rb class Foo @@var = "lion" def self .method01 puts "cat" @name = "cat" @@var = "cat" puts @name end def self .method02 puts "tiger" @name = "tiger" @@var = "tiger" puts @name end def self .method03 puts "dog" @name = "dog" @@var = "dog" puts @name end def putsname puts @name puts @@var end end obj = Foo. new # obj.method01 => (NoMethodError) obj.putsname => lion Foo.method01 Foo.method02 Foo.method03 obj.putsname |
結果:
1
2
3
4
5
6
7
8
9
|
lion cat cat tiger tiger dog dog dog |