Pie*_*rre 6 python abstract-syntax-tree
我正在修改 Python 的ast(抽象语法树)模块。
使用ast.NodeVisitor,可以遍历源树并显示信息,例如每个节点的名称,如下所示:
class MyAST(ast.NodeVisitor):
def visit_ClassDef(self, node):
print(node.col_offset * " " +node.name)
self.generic_visit(node)
def visit_FunctionDef(self, node):
print(node.col_offset * " " + node.name)
self.generic_visit(node)
Run Code Online (Sandbox Code Playgroud)
然后它可以用于任何代码,如下所示:
tree = ast.parse(open("/path/to/file.py").read())
MyAST().visit(tree)
Run Code Online (Sandbox Code Playgroud)
这将输出类似的内容(例如在requests包中的models.py上使用):
RequestEncodingMixin
path_url
_encode_params
_encode_files
RequestHooksMixin
register_hook
deregister_hook
Request
__init__
__repr__
prepare
(...)
Run Code Online (Sandbox Code Playgroud)
我想知道对于给定节点,它的完整“路径”。在上面的输出示例中,当我在FunctionDef节点 中时deregister_hook,我想知道它的父节点,ClassDef RequestHooksMixin以便我可以输出类似 的内容RequestHooksMixin.deregister_hook。
ast.NodeVisitor设计时并未考虑嵌套级别,因此它不是生成基于级别缩进的输出的最佳工具。
相反,您可以ast.iter_child_nodes递归地访问每个节点的直接子节点,并输出该节点并在它属于类或函数时增加缩进级别:
import ast
def build_tree(node):
def _visit(node, level=0):
if isinstance(node, (ast.ClassDef, ast.FunctionDef)):
yield node.name, level
level += 1
for child in ast.iter_child_nodes(node):
yield from _visit(child, level)
return '\n'.join(f'{" " * level}{name}' for name, level in _visit(node))
Run Code Online (Sandbox Code Playgroud)
以便:
source = '''
class Foo:
def method1(self):
pass
def method2(self):
pass
class Bar:
def method1(self):
pass
def method2(self):
pass
'''
print(build_tree(ast.parse(source)))
Run Code Online (Sandbox Code Playgroud)
输出:
Foo
method1
method2
Bar
method1
method2
Run Code Online (Sandbox Code Playgroud)
演示: https: //ideone.com/aQHqbi