今天被問到此類問題,以前總是覺得這個是比較寬泛的一個概念,自己即使是用過這些特性,但卻一直不知道這叫“元編程” 直到今天被人問起的時候,方才頓悟一些,隨后便在網上和自己的平實用的一些元編程做個小總結。
原來所謂的Ruby中的元編程,是可以在運行時動態的操作語言結構(如類、模塊、實例變量等)的技術。你甚至于可以在不用重啟的情況下,在運行時直接鍵入一段新的Ruby代碼,并執行他。
Ruby的元編程,也具有“利用代碼來編寫代碼”的作用。例如,常見的attr_accessor等方法就是如此。
首先我能想到的自己用過的有這么幾個:
1,respond_to?
2,define_method
3,instance_variable_get, instance_variable_set
4,eval 不過這個方法能盡量避免就避免,執行效率特別低!!
具體的意思想必一看方法名字 就知道這些方法的所要實現的邏輯。
具體可詳見http://api.rubyonrails.org/ 上具體分析源碼,很簡單。
以下是我從其他地方搜索而來,以便以后的工作中有助于提高自己的編程技能;
1,內省,反射
在Ruby中,你完全有能力在運行時查看類或對象的信息。我們可以使用class、 instance_methods、 intance_variables等方法來達到目的。我們講這種技術成為內省(Introspection)或者反射(Reflection)。
一說編寫元程序的語言稱之為元語言。被操縱的程序的語言稱之為目標語言。一門編程語言同時也是自身的元語言的能力稱之為反射或者自反。 ——摘自維基百科元編程條目。
復制代碼 代碼如下:
classRubyist
defwhat_does_he_do
@person='A Rubyist'
'Ruby programming'
end
end
an_object = Rubyist.new
puts an_object.class# => Rubyist
puts an_object.class.instance_methods(false)# => what_does_he_do
an_object.what_does_he_do
puts an_object.instance_variables# => @person
respond_to?方法是反射機制中另一個有用的方法。使用respond_to?方法,可以提前知道對象是否能夠處理你想要交與他執行的信息。所有的對象都有此方法,使用respond_to?方法,你可以確定對象是否能使用指定的方法。
2,instance_eval
Object類提供了一個名為instance_eval的公開方法,該方法可被一個實例調用。他提供了操作對象的實例變量的途徑。可以使用字符串向此方法傳遞參數或者傳遞一個代碼塊。
復制代碼 代碼如下:
classRubyist
definitialize
@geek="Matz"
end
end
obj = Rubyist.new
# instance_eval可以操縱obj的私有方法以及實例變量
obj.instance_evaldo
putsself# => #puts@geek# => Matz
end
通過instance_eval傳遞的代碼塊使得你可以在對象內部操作。你可以在對象內部肆意操縱,不再會有任何數據是私有的!instance_eval亦可用于添加類方法。
復制代碼 代碼如下:
classRubyist
end
Rubyist.instance_evaldo
defwho
"Geek"
end
end
puts Rubyist.who# => Geek
const_get, const_set
類似的,const_get和const_set用于操作常量。const_get返回指定常量的值:
復制代碼 代碼如下:
puts Float.const_get(:MIN)# => 2.2250738585072e-308
const_set為指定的常量設置指定的值,并返回該對象。如果常量不存在,那么他會創建該常量,就是下面示范的那樣:
復制代碼 代碼如下:
classRubyist
end
puts Rubyist.const_set("PI",22.0/7.0)# => 3.14285714285714
因為const_get返回常量的值,因此,你可以使用此方法獲得一個類的名字并為這個類添加一個新的實例化對象的方法。這樣使得我們有能力在運行時創建類并實例化其實例。
復制代碼 代碼如下:
# Let us call our new class 'Rubyist'
# (we could have prompted the user for a class name)
class_name ="rubyist".capitalize
Object.const_set(class_name,Class.new)
# Let us create a method 'who'
# (we could have prompted the user for a method name)
class_name =Object.const_get(class_name)
puts class_name# => Rubyist
class_name.class_evaldo
define_method:whodo|my_arg|
my_arg
end
end
obj = class_name.new
puts obj.who('Matz')# => Matz
您可能感興趣的文章:- Ruby元編程的一些值得注意的地方
- ruby元編程之創建自己的動態方法
- ruby元編程之method_missing的一個使用細節
- Ruby元編程之夢中情人method_missing方法詳解
- Ruby元編程技術詳解(Ruby Metaprogramming techniques)
- Ruby和元編程之萬物皆為對象
- ruby元編程實際使用實例
- Ruby元編程基礎學習筆記整理