我对python如何了解包中的模块感到有些困惑。例如,python-pptx
包具有子目录结构为
chart
tree /usr/local/lib/python2.7/site-packages/pptx/chart
/usr/local/lib/python2.7/site-packages/pptx/chart
├── __init__.py
├── axis.py
├── category.py
├── chart.py
├── data.py
├── datalabel.py
├── legend.py
├── marker.py
├── plot.py
├── point.py
├── series.py
├── xlsx.py
└── xmlwriter.py
现在,如果我
import pptx
并将chart
作为
dir(pptx.chart)
然后缺少子模块data
和xlsx
['__builtins__',
'__doc__',
'__file__',
'__name__',
'__package__',
'__path__',
'axis',
'category',
'chart',
'datalabel',
'legend',
'marker',
'plot',
'point',
'series',
'xmlwriter']
我可以使用pptx.chart.axis
直接解决dir(pptx.chart.axis)
,但dir(pptx.chart.data)
会导致
AttributeError: 'module' object has no attribute 'data'
如果我import pptx.chart
但我可以毫无问题地导入pptx.chart.data
。
这里有哪些包导入规则?
Python dir() not displaying all modules in a package
是一个相关的问题,但并没有完全解决我的问题。
正如Alex Hall所解释的那样,一些软件包和模块也会加载其他软件包和模块。例如,使用
python modulefinder.py test.py | grep pptx
我发现,除了许多其他内容之外,还会加载以下内容
P pptx.chart /usr/local/lib/python2.7/site-packages/pptx/chart/__init__.py
m pptx.chart.axis /usr/local/lib/python2.7/site-packages/pptx/chart/axis.py
m pptx.chart.category /usr/local/lib/python2.7/site-packages/pptx/chart/category.py
m pptx.chart.chart /usr/local/lib/python2.7/site-packages/pptx/chart/chart.py
m pptx.chart.datalabel /usr/local/lib/python2.7/site-packages/pptx/chart/datalabel.py
m pptx.chart.legend /usr/local/lib/python2.7/site-packages/pptx/chart/legend.py
m pptx.chart.marker /usr/local/lib/python2.7/site-packages/pptx/chart/marker.py
m pptx.chart.plot /usr/local/lib/python2.7/site-packages/pptx/chart/plot.py
m pptx.chart.point /usr/local/lib/python2.7/site-packages/pptx/chart/point.py
m pptx.chart.series /usr/local/lib/python2.7/site-packages/pptx/chart/series.py
m pptx.chart.xmlwriter /usr/local/lib/python2.7/site-packages/pptx/chart/xmlwriter.py
dir
仅列出已加载的函数,并且由于pptx
未加载pptx.data
,因此dir
不会列出它。要查找所有可能可加载的模块,必须使用help
help(pptx.chart)
Help on package pptx.chart in pptx:
NAME
pptx.chart
FILE
/usr/local/lib/python2.7/site-packages/pptx/chart/__init__.py
PACKAGE CONTENTS
axis
category
chart
data
datalabel
legend
marker
plot
point
series
xlsx
xmlwriter
并且,如果需要来自包的子模块,则应该明确地加载它,而不是依赖于它已被其他东西加载的假设。这是我一直在寻找的规则。
答案 0 :(得分:2)
无论何时导入模块,都会将它们作为属性添加到父模块。如果它们不是从任何地方导入的,它们将无法使用。这是因为创建模块是一个昂贵的过程,您只想创建实际需要的模块。
某处(可能不是由您直接提供,而是由您直接或间接导入的其中一个模块)导入pptx.chart.axis
。
如果您想使用模块,请将其导入以确保它在那里。如果您没有明确导入它,如果它被导入其他地方,它可能仍然存在,但它不可靠。
以下是Django的效果:
>>> import sys
>>> [x for x in sys.modules if x.startswith('django')]
[]
>>> import django
>>> [x for x in sys.modules if x.startswith('django')]
['django.utils.version', 'django', 'django.utils.lru_cache', 'django.utils']
>>> django
<module 'django' from '/Users/alexhall/.pyenv/versions/3.5.1/lib/python3.5/site-packages/django/__init__.py'>
>>> django.utils
<module 'django.utils' from '/Users/alexhall/.pyenv/versions/3.5.1/lib/python3.5/site-packages/django/utils/__init__.py'>
>>> django.core
Traceback (most recent call last):
File "<input>", line 1, in <module>
AttributeError: module 'django' has no attribute 'core'
>>> django.db
Traceback (most recent call last):
File "<input>", line 1, in <module>
AttributeError: module 'django' has no attribute 'db'
>>> import django.db
>>> [x for x in sys.modules if x.startswith('django')]
['django.dispatch', 'django.dispatch.dispatcher', 'django.utils.six.moves', 'django.utils.encoding', 'django.utils.version', 'django.utils.six.moves.urllib', 'django.utils.module_loading', 'django.db', 'django.utils.six', 'django.db.utils', 'django.core.signals', 'django', 'django.utils.functional', 'django.core', 'django.utils.deprecation', 'django.utils.six.moves.urllib.parse', 'django.utils.lru_cache', 'django.conf', 'django.utils.inspect', 'django.utils._os', 'django.core.exceptions', 'django.utils', 'django.conf.global_settings']
>>> django.core
<module 'django.core' from '/Users/alexhall/.pyenv/versions/3.5.1/lib/python3.5/site-packages/django/core/__init__.py'>
>>> django.db
<module 'django.db' from '/Users/alexhall/.pyenv/versions/3.5.1/lib/python3.5/site-packages/django/db/__init__.py'>