Подтвердить что ты не робот

Как аргументы передаются функции через __getattr__

Рассмотрим следующий пример кода (python 2.7):

class Parent:
    def __init__(self, child):
        self.child = child

    def __getattr__(self, attr):
        print("Calling __getattr__: "+attr)
        if hasattr(self.child, attr):
            return getattr(self.child, attr)
        else:
            raise AttributeError(attr)

class Child:
    def make_statement(self, age=10):
        print("I am an instance of Child with age "+str(age))

kid = Child()
person = Parent(kid)

kid.make_statement(5)
person.make_statement(20)

можно показать, что вызов функции person.make_statement(20) вызывает функцию Child.make_statement через функцию Parent __getattr__. В функции __getattr__ я могу распечатать атрибут перед вызовом соответствующей функции в дочернем экземпляре. Пока что так ясно.

Но как передается аргумент вызова person.make_statement(20) через __getattr__? Как я могу распечатать номер "20" в моей функции __getattr__?

4b9b3361

Ответ 1

Вы не печатаете 20 в своей функции __getattr__. Функция находит атрибут make_statement в экземпляре Child и возвращает это. Как это бывает, этот атрибут является методом, поэтому он может быть вызван. Таким образом, Python вызывает возвращаемый метод, и этот метод затем печатает 20.

Если вы хотите удалить вызов (), он все равно будет работать; мы можем сохранить этот метод и вызвать его отдельно, чтобы напечатать 20:

>>> person.make_statement
Calling __getattr__: make_statement
<bound method Child.make_statement of <__main__.Child instance at 0x10db5ed88>>
>>> ms = person.make_statement
Calling __getattr__: make_statement
>>> ms()
I am an instance of Child with age 10

Если вам нужно увидеть аргументы, вам нужно вернуть вместо этого функцию-обертку:

def __getattr__(self, attr):
    print("Calling __getattr__: "+attr)
    if hasattr(self.child, attr):
        def wrapper(*args, **kw):
            print('called with %r and %r' % (args, kw))
            return getattr(self.child, attr)(*args, **kw)
        return wrapper
    raise AttributeError(attr)

Теперь это приводит к:

>>> person.make_statement(20)
Calling __getattr__: make_statement
called with (20,) and {}
I am an instance of Child with age 20