Python包导入:解决子模块的规则是什么?

时间:2017-07-04 20:15:09

标签: python python-packaging python-pptx

我对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)

然后缺少子模块dataxlsx

['__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

这里有哪些包导入规则?

ETA

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

并且,如果需要来自包的子模块,则应该明确地加载它,而不是依赖于它已被其他东西加载的假设。这是我一直在寻找的规则。

1 个答案:

答案 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'>