- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
抱歉,标题令人困惑。至少我试过了!这是我的目录结构:
root\
mypackage\
__init__.py
mymodule.py
main.py
我的模块.py
print('inside mymodule.py')
__init__.py
print('inside __init__')
from . import mymodule as m
主.py
import mypackage
print(mypackage.m)
print(mypackage.mymodule) # <--- Why does it work?
和输出:
inside __init__
inside mymodule.py
<module 'mypackage.mymodule' from 'C:\\Users...\\mypackage\\mymodule.py'>
<module 'mypackage.mymodule' from 'C:\\Users...\\mypackage\\mymodule.py'>
在main.py
文件中,当我导入mypackage
时,这个标签实际上是指__init__.py
文件,所以我可以访问该模块内的所有对象/标签。 mypackage.m
起作用是有道理的,因为 m
现在是 __init__.py
全局命名空间中的一个符号。
但是 __init__.py
的命名空间中没有 mymodule
键/符号,因为我将 mymodule
符号重新绑定(bind)到 m
标签通过 as m
。
问题那么,为什么这个print(mypackage.mymodule)
可以正常工作而不抛出任何异常?
附加信息:如果我在包中有另一个模块,比如说 temp.py
,那么 print(mypackage.temp)
将无法工作,因为 mypackage
引用__init__.py
.
另外对我来说有趣的是,如果我在 __init__.py
中编写 print(mymodule)
并运行 main.py
模块,它将正确运行。
最佳答案
在玩了两天导入语句并搜索文档后,我找到了答案。首先,我要说明发生这种情况的原因,然后再进行更详细的解释。 (另请参阅底部的更新部分)
看看这个link从文档中,还要检查该链接示例中提到的结构树:
If
__all__
is not defined, the statementfrom sound.effects import *
does not import all submodules from the packagesound.effects
into thecurrent namespace; it only ensures that the packagesound.effects
hasbeen imported (possibly running any initialization code in__init__.py
) and then imports whatever names are defined in the package. This includes any names defined (and submodules explicitlyloaded) by__init__.py
.
我们如何在main.py
中导入mypackage
包并不重要,要么是import mypackage
,要么是from mypackage import *
形式无关紧要。
通过这样做,Python 导入了 __init__.py
模块中定义的所有名称,如我们在上面看到的 m
,还有子模块显式加载(此处为 mymodule
模块)。正确地说,它将 'mymodule'
键添加到 __init__.py
的全局命名空间。
我要稍微更改一下 __init__.py
以便我们可以将它直接作为 main 模块运行(因为我们在内部使用了相对导入,所以我们不能这样做)然后我将打印它的全局命名空间中的内容。 (不要忘记用 PYTHONPATH 添加 mypackage
目录)
# __init__.py
print('inside __init__')
from mypackage import mymodule as m
print("--------------------------------")
for k, v in globals().copy().items():
if not k.startswith('__'):
print(k)
输出:
inside __init__
inside __init__
inside mymodule.py
--------------------------------
mymodule
m
--------------------------------
m
你看到 "inside __init__"
打印语句两次,因为这个文件被执行了两次,一次是自己执行,然后是通过执行这一行:from mypackage import mymodule as m
很明显,在虚线下,我们得到了不同的输出。第一个 mymodule
和 m
,第二个只有 m
。
当我们直接运行__init__.py
时,会在sys.modules
中添加一条名为'__main__'
的记录。但是当我们导入 mypackage
时,另一条记录被添加到名为 mypackage
的 sys.modules
中。有趣的是,它们都指向同一位置的同一文件但是从这些文件创建的模块对象并不相同。
为了演示这一点,我将在 mymodule.py
中添加几行代码。这有助于我们查看这些文件和模块:
# mymodule.py
print('inside mymodule.py')
import sys
v1 = sys.modules['mypackage']
v2 = sys.modules['__main__']
print(v1)
print(v2)
print(f'v1 is v2: {v1 is v2}')
print(f'v1 == v2: {v1 == v2}')
print(f'v1.__dict__ == v2.__dict__: {v1.__dict__ == v2.__dict__}')
print('mypackage', list(v1.__dict__))
print('__main__', list(v2.__dict__))
现在让我们再次直接运行__init__.py
模块!
inside __init__
inside __init__
inside mymodule.py
<module 'mypackage' from 'C:\\Users...\\mypackage\\__init__.py'> # these are the same
<module '__main__' from 'C:\\Users...\\mypackage\\__init__.py'> # these are the same
v1 is v2: False
v1 == v2: False
v1.__dict__ == v2.__dict__: False
mypackage ['__name__', '__doc__', '__package__', '__loader__', '__spec__', '__path__', '__file__', '__cached__', '__builtins__']
__main__ ['__name__', '__doc__', '__package__', '__loader__', '__spec__', '__annotations__', '__builtins__', '__file__', '__cached__']
--------------------------------
mymodule
m
--------------------------------
m
如输出所示,它们完全不同。 v1
是包,因为它有 __path__
。
Python 将 'mymodule'
键添加到 __init__.py
的命名空间的唯一情况是我们导入 import mypackage
并且它加载由 __init__.py
显式加载的所有子模块。
我从文档中找到了一个页面,它恰好解决了这个问题:
https://docs.python.org/3/reference/import.html#submodules
When a submodule is loaded using any mechanism, a binding is placed inthe parent module’s namespace to the submodule object.
关于python - 从一个包中访问一个模块,该包的初始文件通过 AS 关键字导入该模块,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/70152714/
如果我创建一个对象时没有使用 new 关键字,例如“Object s(someval)”,但该对象的构造函数使用了 new,当该对象超出范围时,是否会调用析构函数为其分配新的空间?我感觉好像是,但我不
在 SQL 语法中,我发现奇怪的规则表明 select * from ONLY (t1)是有效的 SQL。 我的问题是:什么是 ONLY在这种情况下是什么意思? 它在规范的“7.6 table ref
为什么使用 $(this) 而不是重新选择类很重要? 我在代码中使用了大量的动画和 CSS 编辑,并且我知道可以使用 $(this) 来简化它。 最佳答案 当您通过 jQuery 执行 DOM 查询(
我正在尝试使用 IN 关键字编写查询。 表A 属性标识、属性名称 表B key 、属性标识、属性值 根据提供的 key ,我想返回所有 attrName、attrVal 组合。结果将包含两个表中的列。
这个问题在这里已经有了答案: Why would you use "AS" when aliasing a SQL table? (8 个答案) 关闭 9 年前。 我不擅长写查询,但是从我开始使用
我读过,在 Java 中,您不必将 this 关键字显式绑定(bind)到对象,它由解释器完成。它与 Javascript 相反,在 Javascript 中你总是必须知道 this 的值。但是 Ja
Swift 中“with”关键字的用途是什么?到目前为止,我发现如果您需要覆盖现有的全局函数,例如 toDebugString,可以使用该关键字。 // without "with" you
这个问题在这里已经有了答案: What does the keyword "where" in a class declaration do? (7 个答案) 关闭 9 年前。 在下面的一段代码中(
免责声明:swift 菜鸟 您好,我刚刚开始学习 Swift,正在学习 Swift 编程语言(Apple 在 WWDC 期间发布的书籍),并且想知道“where”关键字是什么。它用于 let vege
深入研究文档后,我找不到以下问题的答案: 是否有任何理由反对使用 this 来引用当前对象,如下例所示? type MyStruct struct { someField string } fun
前言 最近在做THINKPHP开发项目中,用到了 parent:: 关键字,实际上 parent::关键字 是PHP中常要用到的一个功能,这不仅仅是在 THINKPHP 项目开发中,即使是一个小型
我们都知道且经常用到 unsigned 关键字,但有没有想过,与此对应的 signed 关键字有啥用? 复制代码 代码如下: int i = 0; signed
this关键字再java里面是一个我认为非常不好理解的概念,:)也许是太笨的原因 this 关键字的含义:可为以调用了其方法的那个对象生成相应的句柄。 怎么理解这段话呢? thinking i
一 什么是 synchronized synchronized 关键字提供了一种锁机制,能够确保共享变量互斥访问,从而防止数据不一致问题的出现。 synchronized 关键字包括 monitor
最近看了几篇 synchronized 关键字的相关文章,收获很大,想着总结一下该关键字的相关内容。 1、synchronized 的作用 原子性:所谓原子性就是指一个操作或者多个操作,要么全部执行并
在本教程中,您将借助示例了解 JavaScript 对象方法和 this 关键字。 在 JavaScript 中,对象也可以包含函数。例如, // object containing meth
有人可以解释一下 PHP“with”的作用吗? 示例开始: 假设我有一个类: \App\fa_batch 这句话有什么区别: $w = (with (new \App\fa_batch))
这个问题在这里已经有了答案: What is the difference between using the colon and as syntax for declaring type? (2
如果我在 WHERE 子句中使用以下任一项,是否会有很大不同: WHERE [Process Code] = 1 AND ([Material ID] = 'PLT' OR [Material ID]
This question is unlikely to help any future visitors; it is only relevant to a small geographic are
我是一名优秀的程序员,十分优秀!