首先让我们从from import
python 的工作原理开始:
那么首先让我们看一下字节码:
>>> def foo():
... from foo import bar
>>> dis.dis(foo)
2 0 LOAD_CONST 1 (-1)
3 LOAD_CONST 2 (('bar',))
6 IMPORT_NAME 0 (foo)
9 IMPORT_FROM 1 (bar)
12 STORE_FAST 0 (bar)
15 POP_TOP
16 LOAD_CONST 0 (None)
19 RETURN_VALUE
嗯很有趣:),所以from foo import bar
被翻译为 first IMPORT_NAME foo
,相当于import foo
然后IMPORT_FROM bar
。
现在怎么IMPORT_FROM
办?
让我们看看python在他发现时做了什么IMPORT_FROM
:
TARGET(IMPORT_FROM)
w = GETITEM(names, oparg);
v = TOP();
READ_TIMESTAMP(intr0);
x = import_from(v, w);
READ_TIMESTAMP(intr1);
PUSH(x);
if (x != NULL) DISPATCH();
break;
好吧,基本上他得到了要导入的名称,这在我们的foo()
函数 will be 中bar
,然后他从帧堆栈中弹出v
最后一个执行的操作码的返回值 is IMPORT_NAME
,然后import_from()
使用这两个参数调用函数:
static PyObject *
import_from(PyObject *v, PyObject *name)
{
PyObject *x;
x = PyObject_GetAttr(v, name);
if (x == NULL && PyErr_ExceptionMatches(PyExc_AttributeError)) {
PyErr_Format(PyExc_ImportError, "cannot import name %S", name);
}
return x;
}
如您所见,该import_from()
功能很简单,它首先尝试name
从模块中获取属性v
,如果它不存在,则引发ImportError
否则返回此属性。
现在这与相对导入有什么关系?
from . import b
例如,在 OP 问题中的相对导入等价于from pkg import b
.
但这是怎么发生的?要理解这一点,我们应该看一下import.c
python 的模块,特别是函数get_parent()。如您所见,该函数在这里列出的时间很长,但一般来说,当它看到相对导入时,它会.
根据__main__
模块尝试用父包替换点,这再次来自 OP 问题是 package pkg
。
现在让我们把所有这些放在一起,并试图弄清楚为什么我们最终会出现 OP 问题中的行为。
为此,如果我们能看到 python 在导入时做了什么,这将对我们有所帮助,这是我们幸运的一天,python 已经有了这个功能,可以通过在额外详细模式下运行它来启用它-vv
。
所以使用命令行python -vv -c 'import pkg.b'
::
Python 2.6.5 (r265:79063, Apr 16 2010, 13:57:41)
[GCC 4.4.3] on linux2
Type "help", "copyright", "credits" or "license" for more information.
import pkg # directory pkg
# trying pkg/__init__.so
# trying pkg/__init__module.so
# trying pkg/__init__.py
# pkg/__init__.pyc matches pkg/__init__.py
import pkg # precompiled from pkg/__init__.pyc
# trying pkg/b.so
# trying pkg/bmodule.so
# trying pkg/b.py
# pkg/b.pyc matches pkg/b.py
import pkg.b # precompiled from pkg/b.pyc
# trying pkg/a.so
# trying pkg/amodule.so
# trying pkg/a.py
# pkg/a.pyc matches pkg/a.py
import pkg.a # precompiled from pkg/a.pyc
# clear[2] __name__
# clear[2] __file__
# clear[2] __package__
# clear[2] __name__
# clear[2] __file__
# clear[2] __package__
...
Traceback (most recent call last):
File "<string>", line 1, in <module>
File "pkg/b.py", line 1, in <module>
from . import a
File "pkg/a.py", line 2, in <module>
from . import a
ImportError: cannot import name a
# clear __builtin__._
嗯之前发生了ImportError
什么?
首先) from . import a
inpkg/b.py
被调用,如上面解释的那样被翻译成from pkg import a
,在字节码中再次等价于import pkg; getattr(pkg, 'a')
. 但是等一下a
也是一个模块?!那么有趣的部分来了,如果我们from module|package import module
在这种情况下会发生第二次导入,即在 import 子句中导入模块。所以再次在 OP 示例中,我们现在需要导入pkg/a.py
,正如您所知,首先我们sys.modules
为新模块设置密钥,pkg.a
然后我们继续解释模块pkg/a.py
,但在模块pkg/a.py
完成导入之前调用from . import b
.
现在是第二个)部分,pkg/b.py
它将被导入,然后它会首先尝试import pkg
,因为pkg
已经导入,所以我们有一个键pkg
,sys.modules
它只会返回该键的值。然后它将import b
设置pkg.b
密钥sys.modules
并开始解释。我们到达这条线from . import a
!
但请记住pkg/a.py
is 已经导入,这意味着('pkg.a' in sys.modules) == True
导入将被跳过,只会getattr(pkg, 'a')
调用 ,但会发生什么?python没有完成导入pkg/a.py
!?所以 onlygetattr(pkg, 'a')
会被调用,这会AttributeError
在import_from()
函数中引发一个,它会被翻译成ImportError(cannot import name a)
.
免责声明:这是我自己努力了解口译员内部发生的事情,我离成为专家还很遥远。
编辑:这个答案被改写了,因为当我试图再次阅读它时,我注意到我的答案是如何表述错误的,希望现在它会更有用:)