? “Python猫” ,一个值得加星标的公众号

在写 Python 项目的时候,我们可能经常会遇到导入模块失败的错误:ImportError: No module named 'xxx'或者ModuleNotFoundError: No module named 'xxx'

导入失败问题,通常分为两种:一种是导入自己写的模块(即以 .py 为后缀的文件),另一种是导入三方库。本文主要讨论第二种情况,今后有机会,我们再详细讨论其它的相关话题。

解决导入 Python 库失败的问题,其实关键是在运行环境中装上缺失的库(注意是否是虚拟环境),或者使用恰当的替代方案。这个问题又分为三种情况:

一、单个模块中缺失的库

在编写代码的时候,如果我们需要使用某个三方库(如 requests),但不确定实际运行的环境是否装了它,那么可以这样:

try:    import requestsexcept ImportError:    import os    os.system('pip install requests')    import requests

这样写的效果是,如果找不到 requests 库,就先安装,再导入。

在某些开源项目中,我们可能还会看到如下的写法(以 json 为例):

try:    import simplejson as jsonexcept ImportError:    import json

这样写的效果是,优先导入三方库 simplejson,如果找不到,那就使用内置的标准库 json。

这种写法的好处是不需要导入额外的库,但它有个缺点,即需要保证那两个库在使用上是兼容的,如果在标准库中找不到替代的库,那就不可行了。

如果真找不到兼容的标准库,也可以自己写一个模块(如 my_json.py),实现想要的东西,然后在 except 语句中导入它。

try:    import simplejson as jsonexcept ImportError:    import my_json as json

二、整个项目中缺失的库

以上的思路是针对开发中的项目,但是它有几个不足:1、在代码中对每个可能缺失的三方库都 pip install,并不可取;2、某个三方库无法被标准库或自己手写的库替代,该怎么办?3、已成型的项目,不允许做这些修改怎么办?

所以这里的问题是:有一个项目,想要部署到新的机器上,它涉及很多三方库,但是机器上都没有预装,该怎么办?

对于一个合规的项目,按照约定,通常它会包含一个“requirements.txt ”文件,记录了该项目的所有依赖库及其所需的版本号。这是在项目发布前,使用命令pip freeze > requirements.txt 生成的。

使用命令pip install -r requirements.txt (在该文件所在目录执行,或在命令中写全文件的路径),就能自动把所有的依赖库给装上。

但是,如果项目不合规,或者由于其它倒霉的原因,我们没有这样的文件,又该如何是好?

一个笨方法就是,把项目跑起来,等它出错,遇到一个导库失败,就手动装一个,然后再跑一遍项目,遇到导库失败就装一下,如此循环……(此处省略 1 万句脏话)……


三、自动导入任意缺失的库

有没有一种更好的可以自动导入缺失的库的方法呢?

在不修改原有的代码的情况下,在不需要“requirements.txt”文件的情况下,有没有办法自动导入所需要的库呢?

当然有!先看看效果:


我们以 tornado 为例,第一步操作可看出,我们没有装过 tornado,经过第二步操作后,再次导入 tornado 时,程序会帮我们自动下载并安装好 tornado,所以不再报错。

autoinstall 是我们手写的模块,代码如下:

# 以下代码在 python 3.6.1 版本验证通过import sysimport osfrom importlib import import_module

class AutoInstall():    _loaded = set()

    @classmethod    def find_spec(cls, name, path, target=None):            if path is None and name not in cls._loaded:                cls._loaded.add(name)                print("Installing", name)                try:                    result = os.system('pip install {}'.format(name))                    if result == 0:                        return import_module(name)                except Exception as e:                    print("Failed", e)            return None

sys.meta_path.append(AutoInstall)

这段代码中使用了sys.meta_path ,我们先打印一下,看看它是个什么东西?


Python 3 的 import 机制在查找过程中,大致顺序如下:

  • 在 sys.modules 中查找,它缓存了所有已导入的模块
  • 在 sys.meta_path 中查找,它支持自定义的加载器
  • 在 sys.path 中查找,它记录了一些库所在的目录名
  • 若未找到,抛出ImportError异常

其中要注意,sys.meta_path 在不同的 Python 版本中有所差异,比如它在 Python 2 与 Python 3 中差异很大;在较新的 Python 3 版本(3.4+)中,自定义的加载器需要实现find_spec方法,而早期的版本用的则是find_module


以上代码是一个自定义的类库加载器 AutoInstall,可以实现自动导入三方库的目的。需要说明一下,这种方法会“劫持”所有新导入的库,破坏原有的导入方式,因此也可能出现一些奇奇怪怪的问题,敬请留意。

sys.meta_path 属于 Python 探针的一种运用。探针,即import hook,是 Python 几乎不受人关注的机制,但它可以做很多事,例如加载网络上的库、在导入模块时对模块进行修改、自动安装缺失库、上传审计信息、延迟加载等等。

限于篇幅,我们不再详细展开了。最后小结一下:

  • 可以用 try…except 方式,实现简单的三方库导入或者替换
  • 已知全部缺失的依赖库时(如 requirements.txt),可以手动安装
  • 利用 sys.meta_path,可以自动导入任意的缺失库

参考资料: https://github.com/liuchang0812/slides/tree/master/pycon2015cnhttp://blog.konghy.cn/2016/10/25/python-import-hookhttps://docs.python.org/3/library/sys.html#sys.meta_path作者简介:豌豆花下猫,生于广东毕业于武大,现为苏漂程序员,有一些极客思维,也有一些人文情怀,有一些温度,还有一些态度。

优质文章,推荐阅读:

辟谣错误的爬虫说法,使用正确的爬虫姿势为什么range不是迭代器?range到底是什么类型?Python 工匠:使用装饰器的技巧

Python 从业十年是种什么体验?老程序员的一篇万字经验分享

激励猫哥写出更多好文

access 导入txt 找不到可安装的isam_由浅入深:Python 中如何实现自动导入缺失的库?...相关推荐

  1. configure 查找依赖库_由浅入深:Python 中如何实现自动导入缺失的库?

    在写 Python 项目的时候,我们可能经常会遇到导入模块失败的错误:ImportError: No module named 'xxx' 或者 ModuleNotFoundError: No mod ...

  2. 由浅入深:Python 中如何实现自动导入缺失的库?

    在写 Python 项目的时候,我们可能经常会遇到导入模块失败的错误:ImportError: No module named 'xxx' 或者 ModuleNotFoundError: No mod ...

  3. python读取matlab矩阵_matlab、python中矩阵的互相导入导出方式

    还有一种最流行的h5py.. 过几天更新 ------------在python中导出矩阵至matlab------------ 如果矩阵是mxn维的. 那么可以用 : np.savetxt('dev ...

  4. python中模块fitz无法导入成功的原因找到了

    python中模块fitz无法导入成功的原因找到了 python中模块fitz无法导入成功的原因: pip install fitz 成功后,import fitz 出错,模块fitz无法导入成功.其 ...

  5. python中包的循环导入_具有继承的包中的循环导入依赖项

    这是一个可怕的Python循环导入参数,但是,IMHO,您可以有一个优秀的设计,仍然需要循环引用.在 所以,试试这个方法: 在东西.py公司名称:class Thing(Base): def acti ...

  6. python中re模块怎么导入_python如何导入re模块

    正则表达式是一个特殊的字符序列,能方便的检查一个字符串是否与某种模式匹配.re模块使得python拥有全部的正则表达式功能. 正则表达式导入方法import re 在python中默认采用的是贪婪模式 ...

  7. java中的接口自动导入postman和yapi的idea插件----EasyYapi,好用到飞起

    在使用EasyYapi之前我的postman 在使用EasyYapi之后我的postman 接下来是教学: 1.老规矩,idea的Plugin安装插件 安装完成后记得重启哦 2.配置导入的postma ...

  8. 在python中、使用( )语句导入模块_Python导入模块,Python import用法(超级详细)...

    对于一个真实的 Python 程序,我们不可能自己完成所有的工作,通常都需要借助于第三方类库.此外,也不可能在一个源文件中编写整个程序的源代码,这些都需要以模块化的方式来组织项目的源代码. 使用 im ...

  9. 在python中、使用( )语句导入模块_Python import导入模块

    python中的import语句是用来导入模块的,在python模块库中有着大量的模块可供使用,要想使用这些文件需要用import语句把指定模块导入到当前程序中. import语句作用 import语 ...

最新文章

  1. asp.net core 创建允许跨域请求的api, cors.
  2. VC6.0的使用,工程的创建
  3. sqlserver2000导出脚本和导出数据
  4. [经典算法] 河内塔
  5. 手把手教你:亲手打造Silverlight的Win8外观(1) 前言
  6. java的if里有多个if_代码里写很多if会影响效率吗?
  7. pip安装deb_技术|如何在 Ubuntu 上安装 pip
  8. 【bzoj4007】[JLOI2015]战争调度 暴力+树形背包dp
  9. restorecon命令
  10. 下班理财超过上班赚钱
  11. html文字自适应屏幕居中显示,DIV+CSS经典布局[宽度自适应][自动屏幕居中]的实现...
  12. org.apache.catalina.LifecycleException: Protocol handler start failed
  13. HTML学生个人网站作业设计:宠物网站设计——萌宠有家(5页) HTML+CSS 简单DIV布局宠物介绍网页模板代码 DW学生个人网站制作成品下载
  14. 简单使用PHP 的 Silm框架.
  15. IllegalArgumentException: error Type referred to is not an annotation type:
  16. R语言数据可视化之美:专业图表绘制指南
  17. DLM 通讯初始化语句
  18. Linux环境搭建 - update https://apt.repos.intel.com 报错
  19. Win10开始菜单打不开
  20. MINI-UI V3.0 开发框架 非常容易上手

热门文章

  1. new_picview_一款漂亮的图片查看器PictureViewer
  2. oracle pdb是什么意思,浅谈Oracle数据库12c PDB技术
  3. cuda linux编译器_linux下如何编译CUDA+QT(qtcreator下)
  4. php 顺丰接口实例,顺丰BSP接口PHP开发注意事项
  5. 2021 使用IDEA创建并编写你的第一个Maven项目【图文详解】
  6. linux基线核查脚本
  7. 4-什么是Image和container
  8. 【数学】Element Swapping
  9. 问题 A: 深度学习
  10. Reactjs 踏坑指南1: 一些概念