Программирование на языке Ruby
Программирование на языке Ruby читать книгу онлайн
Внимание! Книга может содержать контент только для совершеннолетних. Для несовершеннолетних чтение данного контента СТРОГО ЗАПРЕЩЕНО! Если в книге присутствует наличие пропаганды ЛГБТ и другого, запрещенного контента - просьба написать на почту [email protected] для удаления материала
class MyClass
def self.new_method(name, &block)
define_method(name, &block)
end
end
MyClass.new_method(:mymeth) { puts "Это мой метод." }
x = MyClass.new
x.mymeth # Печатается "Это мой метод."
То же самое можно сделать и на уровне экземпляра, а не класса:
class MyClass
def new_method(name, &block)
self.class.send(:define_method,name, &block)
end
end
x = MyClass.new
x.new_method(:mymeth) { puts "Это мой метод." }
x.mymeth # Печатается "Это мой метод."
Здесь метод экземпляра тоже определен динамически. Изменился только способ реализации метода
new_method
send
define_method
send
По поводу метода
define_method
class MyClass
def self.new_method(name, &block)
define_method(name, &block)
end
end
a,b = 3,79
MyClass.new_method(:compute) { a*b }
x = MyClass.new
puts x.compute # 237
a,b = 23,24
puts x.compute # 552
Смысл здесь в том, что новый метод может обращаться к переменным в исходной области видимости блока, хотя сама эта область более не существует и никаким другим способом не доступна. Иногда это бывает полезно, особенно в случае метапрограммирования или при разработке графических интерфейсов, когда нужно определить методы обратного вызова, реагирующие на события.
Отметим, что замыкание оказывается таковым только тогда, когда имя переменной то же самое. Изредка из-за этого могут возникать сложности. Ниже мы воспользовались методом
define_method
class SomeClass
@@var = 999
define_method(:peek) { @@var }
end
x = SomeClass.new p
x.peek # 999
А теперь попробуем проделать с переменной экземпляра класса такой трюк:
class SomeClass
@var = 999
define_method(:peek) { @var }
end
x = SomeClass.new
p x.peek # Печатается nil
Мы ожидали, что будет напечатано 999, а получили
nil
С другой стороны, такой код работает правильно:
class SomeClass
@var = 999
x = @var
define_method(:peek) { x }
end
x = SomeClass.new p
x.peek # 999
Так что же происходит? Да, замыкание действительно запоминает переменные в текущем контексте. Но ведь контекст нового метода - это контекст экземпляра объекта, а не самого класса.
Поскольку имя
@var
В предыдущих версиях Ruby мы часто определяли методы во время выполнения с помощью
eval
define_method
11.3.6. Метод const_missing
Метод
const_missing
method_missing
Чтобы перехватывать обращения к отсутствующим константам глобально, определите следующий метод в самом классе
Module
Class
class Module
def const_missing(x)
"Из Module"
end
end
class X
end
p X::BAR # "Из Module"
p BAR # "Из Module"
p Array::BAR # "Из Module"
Можно выполнить в нем любые действия: вернуть фиктивное значение константы, вычислить его и т.д. Помните класс
Roman
class Module