Python导入模块时遇到的错误分析


Posted in Python onAugust 30, 2017

当遇到无法导入某个python模块时,可能会是没有安装某个模块,也有可能是某模块在加载过程中失败,也有可能是陷入了循环导入的问题。本文详细解释了这个问题。

1. 模块未安装或者路径不对

ImportError: No mudule named myModule

有两种可能,一是该模块没有安装,一般可以用

pip install %module_name%

来解决。注意有时候模块安装包名并不等于要导入的模块名。这种情况下可以通过pip search | list命令来尝试找到正确的包。

另一种情况就是包虽然安装了,但当前运行的程序加载的路径有错。python运行时将从以下位置尝试加载python modules:

* 当前目录

* 环境变量$PYTHONPATH所指示的值,这是一个由“:”分隔的字符串,各个子字符串都是文件系统的一个路径。

* 标准库目录,如dist-site-packages下的模块。

* 在.pth文件中指定的路径,如果存在.pth文件的话。

可以使用以下方式来查看python运行时的包含路径:

import sys
print(sys.path)

在运行出错的脚本装头部加上这一段代码,然后在控制台中查看打印出来的python类库路径,检查安装包是否已包含在上述路径中。

***可以通过下面的方式将未包含在路径中的模块临时包含进来:***

sys.path.append("path/to/module")

另外,还可以在shell窗口中查看当前的python包含路径:

echo $PYTHONPATH

2. 无法导入已存在的模块

如果要导入的模块包含了native代码,并且native代码加载(初始化)失败时,就会导致这种错误。使用ssl, gevent等涉及native的模块时,如果对应的native程序并未安装,则会出现这样的错误。

另一种错误情况是,使用相对路径导入时,父模块还未导入成功。见下面的代码:

main.py
mypackage/
  __init__.py
mymodule.py
myothermodule.py

mymodule.py如下所示:

#!/usr/bin/env python3

# Exported function
def as_int(a):
  return int(a)

# Test function for module 
def _test():
  assert as_int('1') == 1

if __name__ == '__main__':
  _test()

以及myothermodule代码如下所示:

#!/usr/bin/env python3

from .mymodule import as_int

# Exported function
def add(a, b):
  return as_int(a) + as_int(b)

# Test function for module 
def _test():
  assert add('1', '1') == 2

if __name__ == '__main__':
  _test()

如果执行mypackage/myothermodule,则会报以下错误:

Traceback (most recent call last):
 File "myothermodule.py", line 3, in <module>
   from .mymodule import as_int
SystemError: Parent module '' not loaded, cannot perform relative import
[这篇文章](#Relative imports in Python 3)给出了更详细的解答。

3. 循环导入

这种错误称之为"circular (or cyclic) imports"。是python独有的一种导入错误,在象java这样的语言中就不存在。

假设有如下两个文件,a.py和b.py:

#a.py
print "a in"
import sys
print "b imported: %s" % ("b" in sys.modules, )
import b
print "a out"
print b.x

以及:

#b.py
print "b in"
import a
print "b out"
x = 3

执行python a.py,将得到以下结果:

$ python a.py
a in          
b imported: False
b in
a in
b imported: True
a out
Traceback (most recent call last):
 File "a.py", line 4, in <module>
  import b
 File "/home/shlomme/tmp/x/b.py", line 2, in <module>
  import a
File "/home/shlomme/tmp/x/a.py", line 7, in <module>
  print b.x
AttributeError: 'module' object has no attribute 'x'

出现这种情况的原因是产生了循环导入。循环导入,以及在导入过程中python进行了加锁操作,最终导致在模块b未导入完成时就引用了其中的名字。

判断导入错误是否是因为循环导入引起的,主要看堆栈中是否出现两次重复的导入。比如上述堆栈中a.py出现两次,因此可以判断是这个文件引起的循环导入。

要解决这个问题,可以把模块看成一种资源,对所有要引入的模块进行编号,再按静态资源排序法顺次导入,就可以避免循环导入。

Python 相关文章推荐
深入讲解Python中的迭代器和生成器
Oct 26 Python
Python的re模块正则表达式操作
May 25 Python
Python实现批量压缩图片
Jan 25 Python
Python Numpy 数组的初始化和基本操作
Mar 13 Python
flask框架jinja2模板与模板继承实例分析
Aug 01 Python
Python实现某论坛自动签到功能
Aug 20 Python
python程序 创建多线程过程详解
Sep 23 Python
python 实现list或string按指定分段
Dec 25 Python
Python学习之路之pycharm的第一个项目搭建过程
Jun 18 Python
用python对oracle进行简单性能测试
Dec 05 Python
python实现马丁策略回测3000只股票的实例代码
Jan 22 Python
python opencv旋转图片的使用方法
Jun 04 Python
简单学习Python多进程Multiprocessing
Aug 29 #Python
Python简单实现自动删除目录下空文件夹的方法
Aug 29 #Python
Python实现文件内容批量追加的方法示例
Aug 29 #Python
Python实现解析Bit Torrent种子文件内容的方法
Aug 29 #Python
Python 3.x读写csv文件中数字的方法示例
Aug 29 #Python
在python3环境下的Django中使用MySQL数据库的实例
Aug 29 #Python
Python网络爬虫与信息提取(实例讲解)
Aug 29 #Python
You might like
Windows下的PHP5.0安装配制详解
2006/09/05 PHP
PHP中的加密功能
2006/10/09 PHP
与文件上传有关的php配置参数总结
2013/06/14 PHP
详谈PHP编码转换问题
2015/07/28 PHP
CI框架文件上传类及图像处理类用法分析
2016/05/18 PHP
php json_encode与json_decode详解及实例
2016/12/13 PHP
PHP获取当前执行php文件名的代码
2017/03/02 PHP
PHP简单实现循环链表功能示例
2017/11/10 PHP
PHP进阶学习之依赖注入与Ioc容器详解
2019/06/19 PHP
因str_replace导致的注入问题总结
2019/08/08 PHP
Laravel timestamps 设置为unix时间戳的方法
2019/10/11 PHP
DHTML 中的绝对定位
2006/11/26 Javascript
URL编码转换,escape() encodeURI() encodeURIComponent()
2006/12/27 Javascript
JavaScript 原型链学习总结
2010/10/29 Javascript
JavaScript高级程序设计 阅读笔记(二十一) JavaScript中的XML
2012/09/14 Javascript
jQuery 事件的命名空间简单了解
2013/11/22 Javascript
javascript实现别踩白块儿小游戏程序
2015/11/22 Javascript
5种JavaScript脚本加载的方式
2017/01/16 Javascript
详解微信小程序 wx.uploadFile 的编码坑
2017/01/23 Javascript
tween.js缓动补间动画算法示例
2018/02/13 Javascript
NVM安装nodejs的方法实用步骤
2019/01/16 NodeJs
vue实现多级菜单效果
2019/10/19 Javascript
vue导航栏部分的动态渲染实例
2019/11/01 Javascript
python实现dict版图遍历示例
2014/02/19 Python
Python轻量级ORM框架Peewee访问sqlite数据库的方法详解
2017/07/20 Python
python生成器,可迭代对象,迭代器区别和联系
2018/02/04 Python
Python3从零开始搭建一个语音对话机器人的实现
2019/08/23 Python
Python Django搭建网站流程图解
2020/06/13 Python
毕业生在校学习的自我评价分享
2013/10/08 职场文书
职业生涯规划书的格式
2013/12/29 职场文书
小学学雷锋活动总结
2014/04/25 职场文书
教师节学生演讲稿
2014/09/03 职场文书
一份没有按时交货失信于客户的检讨书
2014/09/19 职场文书
2015年社区矫正工作总结
2015/04/21 职场文书
商业计划书如何写?关键问题有哪些?
2019/07/11 职场文书
TypeScript 内置高级类型编程示例
2022/09/23 Javascript