如果我有一个函数的引用,我可以检查它的代码对象f.__code__
,获取签名,然后对此签名执行后续检查以查看代码是否更改。这很好。
但是,如果其中一个功能的依赖项发生了变化呢? E.g。
def foo(a, b):
return bar(a, b)
假设foo
保持不变,但bar
已更改。有没有办法可以通过foo
对象检查foo.__code__
的“实时”依赖项(与解析文本和使用AST相比)?
答案 0 :(得分:4)
我只是在这里查看__code__
对象,所以我不能说这肯定会起作用,但在我看来你可以使用co_names
元组来(递归地)遍历以特定函数为根的调用图,以便构建可能被调用的函数的传递闭包的某种散列。 (我认为不可能只包含将为特定输入调用的函数,但你可以保守并包含所有可能的调用。)
为此,您需要维护某种符号表,以便能够查找被调用函数的名称。但是一旦你开始沿着这条路走下去,看起来你基本上都会建立自己的AST等价物。那么,为什么不直接使用AST呢?
答案 1 :(得分:1)
您可以使用method.__code__.co_code
比较代码对象上的字节码属性。例如,我们定义两个类:
>>> class A:
... a = 1
... def b(self, b):
... print(self.a + b)
...
>>> class B:
... a = 1
... def b(self, b):
... print(self.a + b)
...
>>> A().b.__code__.co_code
'|\x00\x00j\x00\x00|\x01\x00\x17GHd\x00\x00S'
>>> B().b.__code__.co_code
'|\x00\x00j\x00\x00|\x01\x00\x17GHd\x00\x00S'
>>> A().b.__code__.co_code == B().b.__code__.co_code
True
如果类b
中的方法A
发生了更改:
>>> class A:
... a = 1
... def b(self, b):
... print(b + self.a)
...
>>> A().b.__code__.co_code
'|\x01\x00|\x00\x00j\x00\x00\x17GHd\x00\x00S'
>>> A().b.__code__.co_code == B().b.__code__.co_code
False
或使用检查方法inspect.getsource(object)
:
返回对象的源代码文本。争论可能是 模块,类,方法,函数,回溯,框架或代码对象。 源代码作为单个字符串返回。
如果您想知道代码是否已动态更改,您可能需要使用importlib
重新加载您的类并比较字节码。