我有一个目录,里面填充了任意数量的模块,每个模块只包含一个函数do_stuff
。例如:
foos/
__init__.py
foo_1.py
foo_2.py
...
foo_n.py
例如,foo_1.py可能只是:
def do_stuff(bar):
if bar.blah:
return True
return False
从代码的其他地方开始,我想收集并运行所有do_stuff
函数。我可以使用imp
模块轻松实现这一点:
import imp
...
...
FOOS_DIR = os.path.join(MY_HOME, 'foos/')
MOD_REGEX = r'^(?!__).*\.py$'
foos = [imp.load_source(fname.split('.py')[0], FOOS_DIR)
for fname in os.listdir(FOOS_DIR)
if re.match(MOD_REGEX, fname)]
results = [foo.do_stuff('argument!') for foo in foos]
然而,这导致:
AttributeError: "'module' object has no attribute 'do_stuff'"
从最顶层的回答here,我认为这是要走的路。如果是,也许我的语法是关闭的?毕竟,load_source
真的很宽容:
(pdb) imp.load_source('not_there', FOOS_DIR)
<module 'not_there' from '/home/blah/blah/foos'>
否则,我能错过什么?
答案 0 :(得分:1)
这确实是一个语法问题!我需要到模块文件的完整路径,而不仅仅是它的目录,作为load_source
的第二个参数:
因此,
foos = [imp.load_source(fname.split('.py')[0], FOOS_DIR)
for fname in os.listdir(FOOS_DIR)
if re.match(MOD_REGEX, fname)]
变为
foos = [imp.load_source(fname.split('.py')[0], '%s%s' % (FOOS_DIR, fname))
for fname in os.listdir(FOOS_DIR)
if re.match(MOD_REGEX, fname)]
它运作得很好!