4

我在 Python 中有一堆简单的脚本,它们带有简单的表达式 [1],例如:

C = A+B
D = C * 4

我需要执行它们,但最重要的是我需要知道我依赖的对象是什么;在前一种情况下,对象AB是外部依赖项。例如。鉴于我在一个名为source的 var 中有以前的代码,我希望能够:

deps = { "A" : 1 , "B": 2}
exec source in deps

所以绝对有必要知道如何构建 dict deps

我潜入了ast Python 模块,但我不知道。


[1] 简单的数学聚合,在某种程度上是for循环,仅此而已。

4

1 回答 1

3

您可以使用标准库中的tokenize模块对Python 源代码进行标记。这将允许您找到脚本中使用的所有变量名称。

现在假设我们将“非依赖”定义为紧接在=符号之前的任何变量名。然后,根据您的脚本代码的简单程度(请参阅下面的注意事项),您可以通过这种方式确定不是非依赖项的变量名称:

import tokenize
import io
import token
import collections
import keyword

kwset = set(keyword.kwlist)
class Token(collections.namedtuple('Token', 'num val start end line')):
    @property
    def name(self):
        return token.tok_name[self.num]

source = '''
C = A+B
D = C * 4
'''

lastname = None
names = set()
not_dep = set()
for tok in tokenize.generate_tokens(io.BytesIO(source).readline):
    tok = Token(*tok)
    print(tok.name, tok.val)
    if tok.name == 'NAME':
        names.add(tok.val)
        lastname = tok.val
    if tok.name == 'OP' and tok.val == '=':
        not_dep.add(lastname)

print(names)
# set(['A', 'C', 'B', 'D'])
print(not_dep)
# set(['C', 'D'])

deps = dict.fromkeys(names - not_dep - kwset, 1)
print(deps)
# {'A': 1, 'B': 1}

注意事项

  • 如果您的脚本包含简单赋值以外的语句,则names可能会填充不需要的变量名称。例如,

    import numpy
    

    将两者都添加'import''numpy'集合names中。

  • 如果您的脚本包含使用左侧元组解包的赋值,例如

    E, F = 1, 2
    

    那么上面的幼稚代码只会识别出F它不是依赖项。

于 2013-01-04T22:46:33.433 回答