python中的import,reload,以及__import__

import
作用:
导入/引入一个python标准模块,其中包括.py文件、带有__init__.py文件的目录。
e.g:

[python] view plaincopy

  1. import module_name[,module1,...]
  2. from module import *|child[,child1,...]

说明:
多次重复使用import语句时,不会重新加载被指定的模块,只是把对该模块的内存地址给引用到本地变量环境。
测试:

[python] view plaincopy

  1. a.py
  2. #!/usr/bin/env python
  3. #encoding: utf-8
  4. import os
  5. print ‘in a‘,id(os)
  6. m.py
  7. #!/usr/bin/env python
  8. #encoding: utf-8
  9. import a   #第一次会打印a里面的语句
  10. import os  #再次导入os后,其内存地址和a里面的是一样的,因此这里只是对os的本地引用
  11. print ‘in m‘,id(os)
  12. import a  #第二次不会打印a里面的语句,因为没有重新加载

reload
作用:
对已经加载的模块进行重新加载,一般用于原模块有变化等特殊情况,reload前该模块必须已经import过。
e.g:
import os
reload(os)
说明:
reload会重新加载已加载的模块,但原来已经使用的实例还是会使用旧的模块,而新生产的实例会使用新的模块;reload后还是用原来的内存地址;不能支持from。。import。。格式的模块进行重新加载。
测试:

[python] view plaincopy

  1. a.py
  2. #!/usr/bin/env python
  3. #encoding: utf-8
  4. import os
  5. print ‘in a‘,id(os)
  6. m.py
  7. #!/usr/bin/env python
  8. #encoding: utf-8
  9. import a   #第一次import会打印a里面的语句
  10. print id(a) #原来a的内存地址
  11. reload(a)  #第二次reload还会打印a里面的语句,因为有重新加载
  12. print id(a) #reload后a的内存地址,和原来一样

扩展:
上面说了,在特殊情况的下才会使用reload函数;除了原来模块文件有修改外,还有哪些情况需要使用reload函数呢,这里举个例子。

[python] view plaincopy

  1. #!/usr/bin/env python
  2. #encoding: utf-8
  3. import sys   #引用sys模块进来,并不是进行sys的第一次加载
  4. reload(sys)  #重新加载sys
  5. sys.setdefaultencoding(‘utf8‘)  ##调用setdefaultencoding函数

上面的代码是正确的,再测试下面的代码

[python] view plaincopy

  1. #!/usr/bin/env python
  2. #encoding: utf-8
  3. import sys
  4. sys.setdefaultencoding(‘utf8‘)

上面的测试会失败,那么为什么要在调用setdefaultencoding时必须要先reload一次sys模块呢?因为这里的import语句其实并不是sys的第一次导入语句,也就是说这里其实可能是第二、三次进行sys模块的import,这里只是一个对sys的引用,只能reload才能进行重新加载;那么为什么要重新加载,而直接引用过来则不能调用该函数呢?因为setdefaultencoding函数在被系统调用后被删除了,所以通过import引用进来时其实已经没有了,所以必须reload一次sys模块,这样setdefaultencoding才会为可用,才能在代码里修改解释器当前的字符编码。试试下面的代码,同样会报错:

[python] view plaincopy

  1. #!/usr/bin/env python
  2. #encoding: utf-8
  3. import sys
  4. reload(sys)
  5. sys.setdefaultencoding(‘utf8‘)
  6. del sys.setdefaultencoding   ##删除原来的setdefaultencoding函数
  7. sys.setdefaultencoding(‘gb2312‘)

那么到底是谁在之前就导入sys并且调用了setdefaultencoding函数呢?答案就在python安装目录的Lib文件夹下,有一个叫site.py的文件【python2.6】,在site.py文件里有这么一段代码:

[python] view plaincopy

  1. if hasattr(sys, "setdefaultencoding"):
  2.   del sys.setdefaultencoding

所以在syssetdefaultencoding函数一出来就已经被删除了。

__import__
作用:
同import语句同样的功能,但__import__是一个函数,并且只接收字符串作为参数,所以它的作用就可想而知了。其实import语句就是调用这个函数进行导入工作的,import sys <==>sys = __import__(‘sys‘)
e.g:

__import__(module_name[, globals[, locals[, fromlist]]]) #可选参数默认为globals(),locals(),[]
__import__(‘os‘)    
__import__(‘os‘,globals(),locals(),[‘path‘,‘pip‘])  #等价于from os import path, pip

说明:

通常在动态加载时可以使用到这个函数,比如你希望加载某个文件夹下的所用模块,但是其下的模块名称又会经常变化时,就可以使用这个函数动态加载所有模块了,最常见的场景就是插件功能的支持。

扩展:
既然可以通过字符串来动态导入模块,那么是否可以通过字符串动态重新加载模块吗?试试reload(‘os‘)直接报错,是不是没有其他方式呢?虽然不能直接reload但是可以先unimport一个模块,然后再__import__来重新加载模块。现在看看unimport操作如何实现,在python解释里可以通过globals(),locals(),vars(),dir()等函数查看到当前环境下加载的模块及其位置,但是这些都只能看不能删除,所以无法unimport;不过除此之外还有一个地方是专门存放模块的,这就是sys.modules,通过sys.modules可以查看所有的已加载并且成功的模块,而且比globals要多,说明默认会加载一些额外的模块,接下来就是unimport了。

[python] view plaincopy

    1. #!/usr/bin/env python
    2. #encoding: utf-8
    3. import sys
    4. __import__(‘a‘)      #第一次导入会打印消息
    5. del sys.modules[‘a‘]   #unimport
    6. __import__(‘a‘)    #再次导入还是会打印消息,因为已经unimport一次了
    7. __import__(‘a‘)    #这次就不会打印消息了
时间: 2024-10-23 11:32:39

python中的import,reload,以及__import__的相关文章

python中的import

python中的import操作有点类似于C语言中的#include,但又有很大的不同.在c语言中,#include是编译前将另一个文件包含进当前文件中.在python中导入并非只把一个文件文本插入另一个文件而已. 导入其实是运行时的运算,程序第一次导入时会执行三个步骤. 1 找到模块文件 2 在需要时编译成位码 3执行模块的代码来创建其所定义的对象 这三个步骤只有在模块第一次被导入的时候才会执行,之后导入相同模块的时候会跳过这三个步骤,而只提取内存中所加载的模块对象. python会把载入的模

Python中的import和from import

一.Python路径介绍 在python用import或者from...import来导入相应的模块. 模块其实就是一些函数和类的集合文件,它能实现一些相应的功能,当我们需要使用这些功能的时候,直接把相应的模块导入到我们的程序中,我们就可以使用了. 这类似于C语言中的include头文件,Python中我们用import导入我们需要的模块. 二.搜索路径 Python会在以下路径中搜索它想要寻找的模块: 1. 程序所在的文件夹 2. 标准库的安装路径 3. 操作系统环境变量PYTHONPATH所

浅谈python中得import xxx,from xxx import xxx, from xxx import *

在python中import跟from import都是用来导入的,但是导入的机制不同 1.import xxx:导入模块,或者文件夹,对于调用模块或者文件夹中子模块的变量或者函数,需要使用"模块".XX来调用 2.from xx import xx:导入的是函数或者变量,类,可以直接使用xx 3.from xx import *:导入xx中得全部"公开"变量,属性,公开指的是不是以"_"开头,除非模块或包中的"__all__"

python中的import一个注意事项

1 import math 2 3 def foo(): 4 import math 5 x = math.pi 6 # 如果math在下面import会出错,因为import是个写的过程(添加到sys.modules中),会把math作为 7 # 局部空间变量,上面就找不到了 8 #import math 9 print math.sin(x) 10 11 foo() 上例中函数中的import应该在使用它的地方之前,因为这次import会把math作为局部变量访问 不推荐在文件头把所有的文件

Python中的reload函数

Python中的import语句可以导入module文件,但是import语句只是第一次导入的时候会执行module文件中的代码,然后就会把导入的模块文件存入到内存,当再次导入的时候,Python是直接从内存里面取出module文件,而不会执行module文件的内容了,而reload函数强制Python重新导入并执行module文件. 假设有module文件a.py: def changer(): print("First Version") 然后在可交互的控制台执行如下代码: >

关于python中模块的import路径

前两天被一个同事问了一个python的问题: 为什么一个目录里的python文件引用不要另一个兄弟目录的python文件,但是这两个目录的父母录运行时是可以引用到了.当时感觉一直是模块和包的机制问题,回来翻了一下书,发现是引用路径的问题,记录一下. 在python中我们import模块时,系统会在python的引用路径下去查找这个模块文件, 这个引用路径是在编译或者安装的时候指定的,可以在两个地方进行修改: shell的PYTHONPATH环境变量,系统会在这个变量的目录中进行查找. 在pyth

[C++/Python] 如何在Python中使用一个DLL? (Windows环境)

开发环境VS2012, WIN7 64. 首先生成的DLL大致如下: .h文件 #ifdef CVINPYTHON_EXPORTS #define CVINPYTHON_API __declspec(dllexport) #else #define CVINPYTHON_API __declspec(dllimport) #endif extern "C" { CVINPYTHON_API int addinPython(int, int); }; .c文件 #include &quo

python中import的机制与实现

原文出处: 刘畅(@你猜我猜不猜猜你是谁) 概述 Python 是一门优美简单.功能强大的动态语言.在刚刚接触这门语言时,我们会被其优美的格式.简洁的语法和无穷无尽的类库所震撼.在真正的将python应用到实际的项目中,你会遇到一些无法避免的问题.最让人困惑不解的问题有二类,一个 编码问题,另一个则是引用问题. 本文主要讨论关于Python中import的机制与实现.以及介绍一些有意思的Python Hooks. Python 类库引入机制 首先,看一个简单的例子: """

Ubuntu中python环境下import requests错误的解决(学习过程问题记录)

python中导入模块:import requests提示错误,错误信息描述如下: >>>import requests Traceback (most recent last):   File "<stdin>", line 1, in <module> ImportError:No module named requests 意思就是说没有名为requests的模块,解决方案是在命令行下使用 pip install requests 来进行