python中from module import * 的一个坑

但还有另外一个问题 – 你以为你修改了某个变量,其实,被from module import *后的那个并没有被更新,非常危险,因为程序有可能还可以正常运行, 只不过结果错了,到了production才被发现就比较惨了。

举个例子:

你定义了一些变量在base模块中:

# reference data typeclass Demo: def __init__(self, name):  self.name = namedemo = Demo('Demo')# primitive typefoo = 1

登录后复制

然后在一个模块中用from  module import 的方式读它:

from base import *def read():    print 'reference data id: ' + str(id(demo))    print 'reference data value : ' + demo.name    print 'primitive data id: ' + str(id(foo))    print 'primitive data value: ' + str(foo)

登录后复制

在另外一个模块中写它:

立即学习“Python免费学习笔记(深入)”;

import basedef write(): print "Original:" print "Original reference data id: " + str(id(base.demo)) base.demo.name = "Updated Demo" # this will reflect that change #base.demo = base.Demo("Updated Demo") # this won't relfect the change print "Original data id: " + str(id(base.foo)) base.foo = 1000 print "Original data id after assignment: " + str(id(base.foo))

登录后复制

然后先写,后读,看写的内容是否有效:

import readimport writeprint "before write"read.read()write.write()print "after write"read.read()

登录后复制

结论是没有,原因是:

当你用from module import时,其实是copy了一份reference或者pointer,指向一份内存,var和module.var都指向同一份内存
当你修改module.var时,其实你是让它指向了另外一份内存,此时var和module.var指向的是不同的内存
所以,虽然module.var的值变了,var还是指向原来那份内存,原来的值
这个对于object,比较容易理解,你可以直接修改object里的值,这个是有效的,但是当你指向另外一个object时就无效了。 对于primitive类型来讲,其实也是一个道理,因为每次赋值,都是让其指向一个不同的内存地址,而不是inplace修改已有的那份内存 –  这个很容易验证:

In [1]: a = 10In [2]: id(a)Out[2]: 20429204In [3]: a = 100In [4]: id(a)Out[4]: 20430108

登录后复制

所以,建议是除非是一个quick and dirty的脚本,否则不要使用from module import *!

例子: https://github.com/baiyanhuang/blog/tree/master/arena/python/from_module_import

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至253000106@qq.com举报,一经查实,本站将立刻删除。

发布者:PHP中文网,转转请注明出处:https://www.chuangxiangniao.com/p/2290485.html

(0)
上一篇 2025年2月28日 00:15:59
下一篇 2025年2月28日 00:21:35

AD推荐 黄金广告位招租... 更多推荐

相关推荐

发表回复

登录后才能评论