今天被问到此类问题,以前总是觉得这个是比较宽泛的一个概念,自己即使是用过这些特性,但却一直不知道这叫“元编程” 直到今天被人问起的时候,方才顿悟一些,随后便在网上和自己的平实用的一些元编程做个小总结。
原来所谓的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