автономная функция Python возвращает экземпляр объекта, который вызвал его - PullRequest
0 голосов
/ 10 ноября 2018

У меня очень странная проблема.

Мне нужно вернуть объект класса, чтобы вызвать функцию, которая должна вернуть объект класса, который ее вызвал. Я знаю я знаю. Просто подумайте об этом как обдуманном упражнении, хотя для меня это очень реальная потребность.

def baz():
    # return the object instance that calls me. 

class Foo():
    def bar(self, func):
        return func()  # should return the Foo object but how???

new_foo = Foo()
new_foo_instance = new_foo.bar(baz)

возможно ли написать что-нибудь в baz(), которое будет возвращать объект, который вызвал его?

РЕДАКТИРОВАТЬ:

чтобы ответить на комментарии:

Я пытался использовать inspect, но безуспешно, я даже просмотрел весь стек, но не могу найти запись, соответствующую объекту new_foo:

new_foo выглядит так, когда я его распечатал: <__main__.Foo object at 0x0000029AAFC4C780>

когда я распечатал весь стек, эта запись не была найдена в нем:

def baz():
    print(inspect.stack())
    return inspect.stack() #[1][3]

>>> [FrameInfo(frame=<frame object at 0x0000029AADB49648>, filename='return_caller.py', lineno=5, function='baz', code_context=['    print(inspect.stack())\n'], index=0), FrameInfo(frame=<frame object at 0x0000029AAD8F0DE8>, filename='return_caller.py', lineno=11, function='bar', code_context=['        return func()  # should return the Foo object but how???\n'], index=0), FrameInfo(frame=<frame object at 0x0000029AAD8AC588>, filename='return_caller.py', lineno=19, function='<module>', code_context=['new_foo_instance = new_foo.bar(baz)\n'], index=0)]

Так что я не пытаюсь заставить его вернуть новый экземпляр Foo, но на самом деле точно такой же экземпляр, как new_foo.

Ответы [ 2 ]

0 голосов
/ 10 ноября 2018

Приведенный выше ответ является идеальным, и это еще один способ удовлетворить ваши потребности.

import sys 
import inspect

def baz():
    """
    Return the object instance whose method calls me
    """
    for item in dir(sys.modules[__name__]):
        elem = eval(item)

        if inspect.isclass(elem):
            foo_instance = elem()
            break

    return foo_instance

class Foo():
    """
    Foo class
    """
    def bar(self, func):
        return func()  # should return the Foo object but how???

# Instantiation and calling
new_foo = Foo()
new_foo_instance = new_foo.bar(baz)

print(new_foo_instance)       # <__main__.Foo object at 0x0000015C5A2F59E8>
print(type(new_foo_instance)) # <class '__main__.Fo


# E:\Users\Rishikesh\Projects\Python3\try>python Get_caller_object_final.py
# <__main__.Foo object at 0x0000015C5A2F59E8>
# <class '__main__.Foo'>
Рекомендации "
0 голосов
/ 10 ноября 2018

Используйте осмотреть:

import inspect

def baz():
    frame_infos = inspect.stack()  # A list of FrameInfo.
    frame = frame_infos[1].frame   # The frame of the caller.
    locs = frame.f_locals          # The caller's locals dict.
    return locs['self']

class Foo():
    def bar(self, func):
        return func()

f1 = Foo()
f2 = f1.bar(baz)
print(f1)
print(f2)
print(f2 is f1)  # True

Или обмануть:

def baz():
    return getattr(baz, 'self', None)

class Foo():
    def bar(self, func):
        func.self = self  # Functions can be a place to store global information.
        return func()
...