1

出于调试目的,我们希望获取已知给定对象的所有名称;我想要限定名称(即,A.x而不是x)。这是我们的第一次尝试:

import gc, sys
def find_names(obj):
    result = []
    for referrer in gc.get_referrers(obj):
        if isinstance(referrer, dict):
            for k in referrer:
                if referrer[k] is obj:
                    n = '???'
                    for f in ('__qualname__', '__name__'):
                        if f in referrer:
                            n = referrer[f]
                    result.append('{}.{}'.format(n, k))
    return ','.join(result)

问题是我无法从容器本身找到容器的“名称”,除非容器属于__dict__类或模块(然后__qualname____name__工作)。我不知道为什么__class__不是实例的一部分__dict__(它是否存储在插槽中?),但原因并不重要 - 重要的是我无法得到它。

我能做些什么来使它工作如下:

class A:
    pass
a = A()
a.x = object()
find_names(a.x) # I want this to return `a.x` rather than `???.x`
4

1 回答 1

0

我想我可以通过检查与实例对应的任何引用者的引用者来做到这一点,并寻找__dict__与原始引用者相同的引用者:

def find_names(obj):
    result = []
    for referrer in gc.get_referrers(obj):
        if isinstance(referrer, dict):
            for k in referrer:
                if referrer[k] is obj:
                    n = '???'
                    for f in ('__qualname__', '__name__'):
                        if f in referrer:
                            n = referrer[f]
                            break
                    if n == '???':
                        parents = gc.get_referrers(referrer)
                        for p in parents:
                            if getattr(p, '__dict__', None) is referrer:
                                n = '<instance of class ' + p.__class__.__name__ + '>'
                    result.append(str(n) + '.' + str(k))
    return '\n'.join(sorted(result))
于 2013-05-03T08:28:20.377 回答