解决python3中关于import的疑难杂症

python中import与包管理

概念:模块与包

  • 模块module:一般是以.py为后缀的文件,也包括.pyo.pyc.pyd.so.dll后缀的文件,模块内定义了函数、类以及变量
  • package:包是含有若干个模块的文件夹,在工程项目用包管理模块可以避免模块名冲突

__init__.py

在Python工程项目中,如果一个文件夹下有__init__.py文件就会认为该文件夹是一个包package,这样可以方便组织工程文件,避免模块名冲突。

  • __init__.py为空时仅用于标识当前这个文件夹是一个包package

  • __all__变量指明当该包被import *时,哪些模块module会被导入

  • 可以利用__init__.py对外提供类型、变量及接口,对用户隐藏各个子模块的实现细节

  • 当我们import一个包时,会自动加载该包对应的__init__.py,因此如果在其中做太复杂的运算会造成不必要的开销

sys.modules

sys.modules维护了一个已加载module的字典,第二次加载该module时可以直接从字典中查找,加快执行速度。

import sys
print(sys.modules)

// 输出:
{'random': <module 'random' from '/System/Library/Frameworks/Python.framework/Versions/2.7/lib/python2.7/random.pyc'>, 'subprocess': <module 'subprocess' from '/System/Library/Frameworks/Python.framework/Versions/2.7/lib/python2.7/subprocess.pyc'>, 'sysconfig': <module 'sysconfig' from '/System/Library/Frameworks/Python.framework/Versions/2.7/lib/python2.7/sysconfig.pyc'>, 'gc': <module 'gc' (built-in)>}

namespace

  • local namespace:函数的命名空间,记录函数的变量
  • global namespace:模块的命名空间,记录模块的变量(函数、类、导入的模块、模块级别的变量和常量)
  • build-in namespace:包含build-in functionexceptions,可被任意模块访问

import方式影响我们使用包的方式正是namespace作用的体现:

from foo import bar  # 将模块foo中的函数/变量bar导入到当前模块的命名空间, 可以直接访问bar
import foo  # 导入模块foo同时保留它自己的命名空间, 需要通过foo.bar的方式来访问bar

模块内部属性

  • __doc__:文件注释
  • __file__:当前文件路径
  • __package__:导入文件的路径
  • __cached__:导入文件的缓存路径
  • __name__:导入文件的路径加文件名称
  • __builtins__:包含内置函数

python内置模块

  • os:提供文件和目录等的系统级操作
  • sys:提供对解释器相关的操作
  • hashlib:提供加密相关的操作,替代了md5sha模块
  • shutil:提供文件、文件夹和压缩包等处理模块
  • configparser:提供对特定配置的操作
  • logging:提供日志功能
  • timedatetime:提供时间相关操作
  • random:提供随机数操作
  • jsonpickle:提供序列化操作
  • shelve:提供简单kv将内存数据通过文件持久化的功能

import方式

1. 简介

在Python中import的常用操作为:

import somemodule  # 导入整个模块
from somemodule import somefunction  # 从模块中导入单个函数
from somemodule import firstfunc, secondfunc, thirdfunc  # 从模块中导入多个函数
from somemodule import *  # 从模块中导入所有函数

2. 执行import的步骤

  1. 创建一个新的module对象
  2. 将该module对象插入sys.modules
  3. 装载module的代码
  4. 执行新的module中对应的代码

3. import的搜索包顺序

注意第三步装载module代码时python解释器需要先搜索到对应的.py文件,搜索顺序为:

  • sys.path:包含了当前脚本的路径和其他查找包(系统库、第三方库等)的路径,你也可以在代码中通过sys.path.append()动态添加搜索路径
  • PYTHONPATH
  • 查看默认路径,比如Linux下为/usr/local/lib/python/

4. 绝对导入与相对导入

绝对导入和相对导入的概念只针对于包内模块导入包内模块,注意如果foo.pybar.py在同一个非包(没有__init__.py文件)的目录下,那么它们之间可以互相import,不存在绝对导入和相对导入的问题。

在Python3中建议使用绝对导入。

举个例子:

$ tree
mypackage
├── __init__.py
├── module_bar.py
└── module_foo.py

在包mypackage内,如果module_bar要导入module_foo,那么有三种方式:

# 方法一: 
import module_foo

# 方法二:
# 如果是上层文件夹写.., 上上层文件夹写..., 以此类推
from . import module_foo

# 方法三:
from mypackage import module_foo
import mypackage.module_foo
  • 对于python2而言,方法一和方法二都是相对导入,效果一样,但是前者被称为隐式相对导入,后者被称为显式相对导入,方法三是绝对导入(会在sys.path中的路径搜索)

  • 对于python3而言,方法二是相对导入,方法一和方法三都是绝对导入,官方更推荐方法三

5. 包导入

包的导入和模块导入基本一致,只不过导入包时会执行__init__.py。如果只是导入一个包import package而不指名任何模块,且包中的__init__.py没有其他的初始化操作,那么包下面的模块是无法被自动导入的。

6. 直接运行与模块运行

以下面的项目为例:

$ tree
.
└── mypackage
    ├── __init__.py
    └── module_foo.py
    
# module_foo.py内容如下:
import sys
print(sys.path)

我们有两种方式运行module_foo.py

-m参数表示run library module as a script,即以脚本的方式执行模块。

# 直接运行: 第一个目录是模块module_foo所在的
$ python3 -B mypackage/module_foo.py    
['/Users/didi/Desktop/MyProject/mypackage', '/Library/Frameworks/Python.framework/Versions/3.7/lib/python37.zip', '/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7', '/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/lib-dynload', '/Users/didi/Library/Python/3.7/lib/python/site-packages', '/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages']

# 模块运行: 第一个目录是当前路径
$ python3 -B -m mypackage.module_foo
['/Users/didi/Desktop/MyProject', '/Library/Frameworks/Python.framework/Versions/3.7/lib/python37.zip', '/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7', '/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/lib-dynload', '/Users/didi/Library/Python/3.7/lib/python/site-packages', '/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages']

实例:包之间模块引用的疑难杂症

1. 项目demo

假设当前你的工程文件目录如下(仅针对python3):

注意这里我的文件夹下并没有__init__.py,严格来讲它们并不是包,只是将联系紧密的模块放在同一个文件夹下方便工程项目管理。

.
└── src
    ├── bar_package
    │   └── module_1.py
    ├── foo_package
    │   ├── module_2.py
    │   └── module_3.py
    └── main.py
# 注意
# 1) 所有模块都以src为根目录, 包括main.py(当然这只是我个人习惯)
# 2) 引入方式都是绝对引入(python3推荐使用)

"""
module_1.py: 空文件
"""

"""
module_2.py: import同个包内的module_3
"""
from foo_package import module_3  # 引用同个包的模块

"""
module_3.py: import另一个包内的module_1
"""
from bar_package import module_1  # 跨包引用模块

if __name__ == "__main__":
    print("module_3 exec successfully!")

"""
main.py: import所有模块
"""
from foo_package import module_3, module_2
from bar_package import module_1

上面就是通常项目文件包管理的方式,执行整个程序:

python3 -B src/main.py

2. 问题:单独执行某个模块

如果要单独执行module_3.py,这时候会报错:

$ python3 -B src/foo_package/module_3.py 
Traceback (most recent call last):
  File "src/foo_package/module_3.py", line 1, in <module>
    from bar_package import module_1  # 跨包引用模块
ModuleNotFoundError: No module named 'bar_package'

回顾一下之前提到的import查找包的路径,我们有两种方法可以解决这个问题。

3. 方法一:通过模块运行的方式解决(推荐)

本质上我们是希望将module_3.py这个模块作为脚本运行,所以我们可以带上-m参数:

$ cd src  # 代码中是以src为根目录的, 所以需要进入到src下
$ python3 -B -m foo_package.module_3   
module_3 exec successfully!

4. 方法二:在sys.path中添加查找路径

前面的报错是找不到bar_package的模块名,因为直接运行的话sys.path第一个路径就是module_3.py的路径,自然找不到它上层的bar_package,我们可以通过sys.path.append(..)将它的上层目录也加入sys.path,修改后的module_3.py文件内容为:

"""module_3.py
本质上就是将module_3.py的上级目录加入到sys.path中, 这样就可以找到bar_package了
"""
import os
import sys
parent_path = os.path.dirname(sys.path[0])
if parent_path not in sys.path:
    sys.path.append(parent_path)
from bar_package import module_1  # 跨包引用模块


if __name__ == "__main__":
    print("module_3 exec successfully!")

需要注意的是,如果你使用的是如下这种写法还是可能出现问题:

"""module_3.py
"""
import sys
sys.path.append("../")
from bar_package import module_1  # 跨包引用模块


if __name__ == "__main__":
    print("module_3 exec successfully!")

# 进入到module_3.py所在的目录, 输出正常:
$ src/foo_package 
$ python3 -B module_3.py 
module_3 exec successfully!

# 直接在根目录下执行会报错:
$ python3 -B src/foo_package/module_3.py
Traceback (most recent call last):
  File "src/foo_package/module_3.py", line 3, in <module>
    from bar_package import module_1  # 跨包引用模块
ModuleNotFoundError: No module named 'bar_package'

另一种简洁的写法是:

import sys
import oss
sys.path.append(os.path.dirname(os.path.dirname(os.path.realpath(__file__))))

5. 尽量不要使用相对引用

Python3不建议使用相对引用,最好遵循一定的开发规范,不要在代码中混用绝对引用与相对引用。

Reference

[1] https://blog.csdn.net/weixin_38256474/article/details/81228492

[2] https://zhuanlan.zhihu.com/p/115350758

[3] https://www.cnblogs.com/schips/p/12148092.html

[4] https://www.jianshu.com/p/88b0f6f28f25

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,657评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,662评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,143评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,732评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,837评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,036评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,126评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,868评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,315评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,641评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,773评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,859评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,584评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,676评论 2 351

推荐阅读更多精彩内容

  • 当前目录 和 脚本目录 参考资料:https://techibee.com/python/get-current-...
    ThomasYoungK阅读 10,869评论 0 11
  • 1. 标准 import Python 中所有加载到内存的模块都放在 sys.modules 。当 import ...
    唐文阁阅读 1,736评论 0 1
  • 在前面的几个章节中我们脚本上是用 python 解释器来编程,如果你从 Python 解释器退出再进入,那么你定义...
    Java丶python攻城狮阅读 293评论 0 0
  • 防忘系列... Pycharm并没有IDEA一样可以设置对import进行排序以及自动消除没用到的库(要用快捷键C...
    我只要喝点果粒橙阅读 1,702评论 0 1
  • 用 python 解释器来编程从 Python 解释器退出再进入,那么你定义的所有的方法和变量就都消失了。 为此...
    chen_000阅读 521评论 0 3