所以,我有一个小小的Python导入的谜团。我相信由于某种原因应该是这样的,因为Guido很少犯错。但是,为什么会这样呢?
$ cat myModule.py
#!/usr/bin/python
class SomeModule(object):
def __init__(self):
print "in SomeModule.__init__ ! "
def doSomething(self):
print 'doing something.'
$ cat myTest.py
import unittest
from myModule import SomeModule
class TestMyModule(unittest.TestCase):
def test_001(self):
print "should see init below"
sm = SomeModule()
sm.doSomething()
print "should see init above\n"
def test_002(self):
print "should not see init below."
from myModule import SomeModule as SM2
SM2.__init__ = lambda x: None
sm2 = SM2()
sm2.doSomething()
print "should not have seen init above.\n"
def test_bbb(self):
print "Should see init below"
sm = SomeModule()
sm.doSomething()
print "should see init above\n"
$ nosetests myTest.py -s
should see init below
in SomeModule.__init__ !
doing something.
should see init above
.should not see init below.
doing something.
should not have seen init above.
.Should see init below
doing something.
should see init above
.
----------------------------------------------------------------------
Ran 3 tests in 0.001s
OK最后的测试应该不受中间测试的影响,对吗?我的直觉说,我不必担心最初的导入,因为第二个导入使用'as‘。因此,在最后一次测试中,我希望看到init,但我没有.。
显然,第二个导入,‘从myModule导入SomeModule作为SM2',破坏了某些模块的初始导入,尽管它看起来应该是一个完全独立的实体,如SM2,而不是SomeModule。
这对某人来说有意义吗?
发布于 2014-02-04 17:03:20
这里的“问题”是SM2和SomeModule是相同的类。在python中,每个模块只有一个“实例”--重新启用将返回对同一个对象的引用:
>>> import code
>>> import code as code2
>>> code is code2
True在您的代码中,当您运行SM2.__init__ = lambda x: None时,您将替换该类的构造函数。这会影响SM2、SomeModule以及可能从myModule导入的任何其他代码。
如果简单地使用SM2= SomeModule对类进行别名化,也会发生同样的事情。
当新程序员开始处理列表并期望分配给一个新变量以复制实际列表时,同样的机制也会让他们感到惊讶:
>>> a=[1]
>>> b=a
>>> b.append(2)
>>> print a
[1, 2]可以使用is验证代码中对象的标识。注意,这与相等并不相同。
>>> []==[]
True
>>> [] is []
Falsehttps://stackoverflow.com/questions/21558543
复制相似问题