Метод remove_constудаляет константу.
module Math
remove_const :PI
end
# PI больше нет!
Отметим, что таким способом можно удалить и определение класса (потому что идентификатор класса — это просто константа):
class BriefCandle
#...
end
out_out = BriefCandle.new
class Object
remove_const :BriefCandle
end
# Создать еще один экземпляр класса BriefCandle не получится!
# (Хотя out_out все еще существует...)
Такие методы, как remove_constи remove_method, являются закрытыми (что и понятно). Поэтому во всех примерах они вызываются изнутри определения класса или модуля, а не снаружи.
11.3.8. Получение списка определенных сущностей
API отражения в Ruby позволяет опрашивать классы и объекты во время выполнения. Рассмотрим методы, имеющиеся для этой цели в Module, Classи Object.
В модуле Moduleесть метод constants, который возвращает массив всех констант, определенных в системе (включая имена классов и модулей). Метод nestingвозвращает массив всех вложенных модулей, видимых в данной точке программы.
Метод экземпляра Module#ancestorsвозвращает массив всех предков указанного класса или модуля.
list = Array.ancestors
# [Array, Enumerable, Object, Kernel]
Метод constantsвозвращает список всех констант, доступных в данном модуле. Включаются также его предки.
list = Math.constants # ["E", "PI"]
Метод class_variablesвозвращает список всех переменных класса в данном классе и его суперклассах. Метод included_modulesвозвращает список модулей, включенных в класс.
class Parent
@@var1 = nil
end
class Child < Parent
@@var2 = nil
end
list1 = Parent.class_variables # ["@@var1"]
list2 = Array.included_modules # [Enumerable, Kernel]
Методы instance_methodsи public_instance_methodsкласса Class— синонимы; они возвращают список открытых методов экземпляра, определенных в классе. Методы private_instance_methodsи protected_instance_methodsведут себя аналогично. Любой из них принимает необязательный булевский параметр, по умолчанию равный true; если его значение равно false, то суперклассы не учитываются, так что список получается меньше.
n1 = Array.instance_methods.size # 121
n2 = Array.public_instance_methods.size # 121
n3 = Array.private_instance_methods.size # 71
n4 = Array.protected_instance_methods.size # 0
n5 = Array.public_instance_methods(false).size # 71
В классе Objectесть аналогичные методы, применяющиеся к экземплярам (листинг 11.17). Метод methodsвозвращает список всех методов, которые можно вызывать для данного объекта. Метод public_methodsвозвращает список открытых методов и принимает параметр, равный по умолчанию true, который говорит, нужно ли включать также методы суперклассов. Методы private_methods, protected_methodsи singleton_methodsтоже принимают такой параметр.
Листинг 11.17. Отражение и переменные экземпляра
class SomeClass
def initialize
@a = 1
@b = 2
end
def mymeth
# ...
end
protected :mymeth
end
x = SomeClass.new
def
x.newmeth
# ...
end
iv = x.instance_variables # ["@b", "@a"]
p x.methods.size # 42
p x.public_methods.size # 41
p x.public_methods(false).size # 1
p x.private_methods.size # 71
p x.private_methods(false).size # 1
p x.protected_methods.size # 1
p x.singleton_methods.size # 1
Если вы работаете с Ruby уже несколько лет, то заметите, что эти методы немного изменились. Теперь параметры по умолчанию равны true, а не false.
11.3.9. Просмотр стека вызовов
And you may ask yourself:
Well, how did I get here? [13] И задаешь себе вопрос: Как же я оказался здесь?
Talking Heads, «Once in a Lifetime»
Иногда необходимо знать, кто вызвал метод. Эта информация полезна, если, например, произошло неисправимое исключение. Метод caller, определенный в модуле Kernel, дает ответ на этот вопрос. Он возвращает массив строк, в котором первый элемент соответствует вызвавшему методу, следующий — методу, вызвавшему этот метод, и т.д.
def func1
puts caller[0]
end
def func2
func1
end
func2 # Печатается: somefile.rb:6:in 'func2'
Строка имеет формат «файл;строка» или «файл;строка в методе».
11.3.10. Мониторинг выполнения программы
Программа на Ruby может следить за собственным выполнением. У этой возможности есть много применений; интересующийся читатель может заглянуть в исходные тексты программ debug.rb, profile.rbи tracer.rb. С ее помощью можно даже создать библиотеку для «проектирования по контракту» (design-by-contract, DBC), хотя наиболее популярная в данный момент библиотека такого рода этим средством не пользуется.
Читать дальше
Конец ознакомительного отрывка
Купить книгу