栏目分类:
子分类:
返回
名师互学网用户登录
快速导航关闭
当前搜索
当前分类
子分类
实用工具
热门搜索
名师互学网 > IT > 面试经验 > 面试问答

Python中的循环模块依赖性和相对导入

面试问答 更新时间: 发布时间: IT归档 最新发布 模块sitemap 名妆网 法律咨询 聚返吧 英语巴士网 伯小乐 网商动力

Python中的循环模块依赖性和相对导入

首先让我们从

from import
python的工作方式开始:

首先让我们看一下字节码:

>>> def foo():...     from foo import bar>>> dis.dis(foo)20 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
先翻译
import_NAME foo
importfoo
,然后翻译为
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()
函数中将是
bar
,然后他从帧堆栈中弹出一个值
v
,该值是最后执行的操作码的返回值
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 linux2Type "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__.pyimport pkg # precompiled from pkg/__init__.pyc# trying pkg/b.so# trying pkg/bmodule.so# trying pkg/b.py# pkg/b.pyc matches pkg/b.pyimport 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.pyimport 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 aimportError: cannot import name a# clear __builtin__._

嗯,之前发生了

importError
什么?

首先) 调用

from . import a
in
pkg/b.py
,按上面的解释将其翻译为
from pkg importa
,再次以字节码表示相当于
import pkg; getattr(pkg,'a')
。但是等一下
a
模块也是吗?!好了,这是有趣的部分,如果我们有类似
from module|package importmodule
的情况,在这种情况下,将进行第二次导入,即import子句中模块的导入。因此,再次在OP示例中,我们现在需要导入
pkg/a.py
,并且您首先知道,我们
sys.modules
为新模块设置了一个密钥,该密钥将是
pkg.a
,然后我们继续对模块进行解释
pkg/a.py
,但是在模块
pkg/a.py
完成导入之前调用
from. import b

现在进入 第二 部分,

pkg/b.py
将被导入,然后它将首先尝试
importpkg
,因为
pkg
已经被导入,所以我们中有一个键
pkg
sys.modules
它将只返回该键的值。然后它将
importb
设置
pkg.b
密钥
sys.modules
并开始解释。我们到达这条线
from . import a

但是 请记住

pkg/a.py
已经导入了,这意味着
('pkg.a' in sys.modules) ==True
导入将被跳过,仅
getattr(pkg,'a')
将调用,但是会发生什么呢?python尚未完成导入
pkg/a.py
!因此,只会
getattr(pkg,'a')
调用,这将
AttributeError
import_from()
函数中引发,并将其转换为
importError(cannotimport name a)

免责声明 :这是我自己的工作,以了解口译员内部发生的事情,我远不是专家。

EDIt: 改写了这个答案,因为当我再次尝试阅读它时,我指出了我的答案是不好的,希望现在它会更有用:)



转载请注明:文章转载自 www.mshxw.com
本文地址:https://www.mshxw.com/it/637994.html
我们一直用心在做
关于我们 文章归档 网站地图 联系我们

版权所有 (c)2021-2022 MSHXW.COM

ICP备案号:晋ICP备2021003244-6号