Три пути: send
/ call
/ eval
- и их контрольные показатели
Типичный вызов (для справки):
s= "hi man"
s.length #=> 6
С помощью send
s.send(:length) #=> 6
С помощью call
method_object = s.method(:length)
p method_object.call #=> 6
С помощью eval
eval "s.length" #=> 6
Ориентиры
require "benchmark"
test = "hi man"
m = test.method(:length)
n = 100000
Benchmark.bmbm {|x|
x.report("call") { n.times { m.call } }
x.report("send") { n.times { test.send(:length) } }
x.report("eval") { n.times { eval "test.length" } }
}
... как видите, создание экземпляра объекта метода - это самый быстрый динамический способ вызова метода, также обратите внимание на то, насколько медленно работает eval.
#######################################
##### The results
#######################################
#Rehearsal ----------------------------------------
#call 0.050000 0.020000 0.070000 ( 0.077915)
#send 0.080000 0.000000 0.080000 ( 0.086071)
#eval 0.360000 0.040000 0.400000 ( 0.405647)
#------------------------------- total: 0.550000sec
# user system total real
#call 0.050000 0.020000 0.070000 ( 0.072041)
#send 0.070000 0.000000 0.070000 ( 0.077674)
#eval 0.370000 0.020000 0.390000 ( 0.399442)
Авторы этой статьи пишут в блоге, в которой более подробно рассматриваются три метода, а также показано, как проверить, существуют ли эти методы.