设为首页 收藏本站
查看: 670|回复: 0

[经验分享] 【Python学习笔记】1. import reload 以及__import__注意点

[复制链接]

尚未签到

发表于 2015-4-22 11:57:27 | 显示全部楼层 |阅读模式
  import
作用:
导入/引入一个python标准模块,其中包括.py文件、带有__init__.py文件的目录
  说明:



1 import module_name[,module1,...]  
2 from module import *|child[,child1,...]  
  
  多次重复使用import语句时,不会重新加载被指定的模块,只是把对该模块的内存地址给引用到本地变量环境。



1 a.py  
2 #!/usr/bin/env python   
3 #encoding: utf-8  
4 import os  
5 print 'in a',id(os)  
6   
7 m.py  
8 #!/usr/bin/env python   
9 #encoding: utf-8  
10 import a   #第一次会打印a里面的语句  
11 import os  #再次导入os后,其内存地址和a里面的是一样的,因此这里只是对os的本地引用  
12 print 'in c',id(os)  
13 import a  #第二次不会打印a里面的语句,因为没有重新加载  
  
  reload
作用:
对已经加载的模块进行重新加载,一般用于原模块有变化等特殊情况,reload前该模块必须已经import过。
  e.g:
import os
reload(os)
说明:
reload会重新加载已加载的模块,但原来已经使用的实例还是会使用旧的模块,而新生产的实例会使用新的模块;reload后还是用原来的内存地址;不能支持from。。import。。格式的模块进行重新加载。
  



1 a.py  
2 #!/usr/bin/env python   
3 #encoding: utf-8  
4 import os  
5 print 'in a',id(os)  
6   
7 m.py  
8 #!/usr/bin/env python   
9 #encoding: utf-8  
10 import a   #第一次import会打印a里面的语句  
11 print id(a) #原来a的内存地址  
12 reload(a)  #第二次reload还会打印a里面的语句,因为有重新加载  
13 print id(a) #reload后a的内存地址,和原来一样  
  上面说了,在特殊情况的下才会使用reload函数;除了原来模块文件有修改外,还有哪些情况需要使用reload函数呢,这里举个例子。



1 #!/usr/bin/env python   
2 #encoding: utf-8  
3 import sys   #引用sys模块进来,并不是进行sys的第一次加载  
4 reload(sys)  #重新加载sys  
5 sys.setdefaultencoding('utf8')  ##调用setdefaultencoding函数  
  上面的代码是正确的,再测试下面的代码



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才会为可用,才能在代码里修改解释器当前的字符编码。试试下面的代码,同样会报错:



#!/usr/bin/env python   
#encoding: utf-8  
import sys   
reload(sys)   
sys.setdefaultencoding('utf8')   
del sys.setdefaultencoding   ##删除原来的setdefaultencoding函数     
sys.setdefaultencoding('gb2312')  

  那么到底是谁在之前就导入sys并且调用了setdefaultencoding函数呢?答案就在python安装目录的Lib文件夹下,有一个叫site.py的文件【python2.6】,在里面可以找到main() --> setencoding()-->sys.setdefaultencoding(encoding),因为这个site.py每次启动python解释器时会自动加载,所以main函数每次都会被执行,setdefaultencoding函数一出来就已经被删除了。
  
  __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了。



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')    #这次就不会打印消息了
  __import__可以把字符串形式的模块进行导入,但有一点要注意,你需要让python知道如何找到这个模块,因此这个模块的路径可能需要加到 sys.path 中去。那么在我写的meteor模块处理模块中我遇到一个问题,当python格式的模板(它就是使用__import__来导入模板的)被导入后,然后进行了修改,如何重新导入使其生效呢。那么就可以使用reload来实现。代码如下:



1         dirname = os.path.dirname(os.path.abspath(modulename))
2         filename, ext = os.path.splitext(os.path.basename(modulename))
3         if ext.lower() != ‘.py’:
4             return {}, {}
5         if dirname:
6             sys.path.insert(0, dirname)
7         if sys.modules.has_key(filename):
8             mod = sys.modules[filename]
9             reload(mod)
10         else:
11             mod = __import__(filename)
12         if dirname:
13             del sys.path[0]
  这段代码是用来导入一个python的文件,你传给一个字符串格式的python文件名,如:template/tmpl.py。如果有目录则将其插入到sys.path中,最后再删除。如果在sys.modules已经存在此模块,说明以前导入过,则调用reload重新装入,否则是一个未导入过的模块,可以使用__import__进行导入。
  问题就是原来代码不是这样写的,可能是这样:




1         dirname = os.path.dirname(os.path.abspath(modulename))
2         filename, ext = os.path.splitext(os.path.basename(modulename))
3         if ext.lower() != ‘.py’:
4             return {}, {}
5         if sys.modules.has_key(filename):
6             mod = sys.modules[filename]
7             reload(mod)
8         else:
9             if dirname:
10                 sys.path.insert(0, dirname)
11             mod = __import__(filename)
12             if dirname:
13                 del sys.path[0]
  
  注意背影为红色的代码,也就是说在调用reload()之前没有对路径的处理。这样做报了一个错,说是找不到模块。因此才改成开始的代码。
  那么这个问题就是说,因为reload()与__import__差不多都需要在sys.path中可以找到模块,因此都需要把路径加进去。尽管你可以从sys.modules中得到模块对象,但reload()是需要从源码重新导入的,依然需要通过sys.path来导入模块。
  不过,这段代码最后变成了:



1  dirname = os.path.dirname(os.path.abspath(modulename))
2         filename, ext = os.path.splitext(os.path.basename(modulename))
3         if ext.lower() != ‘.py’:
4             return {}, {}
5         if sys.modules.has_key(filename):
6             del sys.modules[filename]
7         if dirname:
8             sys.path.insert(0, dirname)
9         mod = __import__(filename)
10         if dirname:
11             del sys.path[0]
  没有使用reload()不过功能是一样的。
  

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-59602-1-1.html 上篇帖子: Python列表copy 下篇帖子: python转载[编码问题]
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表