mock.MagicMock(spec=SomeClass) 和 mock.create_autospec(SomeClass) 有什么区别?
What's the difference between mock.MagicMock(spec=SomeClass) and mock.create_autospec(SomeClass)?
我正在尝试了解这两种模拟结构之间的区别以及何时适合使用它们。我在解释器中测试过,例如:
>>> mm = mock.MagicMock(spec=list)
>>> ca = mock.create_autospec(list)
>>> mm
<MagicMock spec='list' id='140372375801232'>
>>> mm()
<MagicMock name='mock()' id='140372384057808'>
>>> mm.append()
<MagicMock name='mock.append()' id='140372375724720'>
>>> mm().append()
<MagicMock name='mock().append()' id='140372375753104'>
>>> ca
<MagicMock spec='list' id='140372384059248'>
>>> ca()
<NonCallableMagicMock name='mock()' spec='list' id='140372384057040'>
>>> ca.append()
<MagicMock name='mock.append()' id='140372375719744'>
>>> ca().append()
<MagicMock name='mock().append()' id='140372375796848'>
>>>
但我不明白为什么“构建”使用 create_autospec
创建的模拟给我 NonCallableMagicMock
而 MagicMock
给了我更多 MagicMock
。 documentation 没有多大帮助。
使用 spec
参数和使用 create_autospec
的主要区别在于递归性。第一种情况,对象本身是指定的,而被调用的对象不是:
>>> mm = mock.MagicMock(spec=list)
>>> mm
<MagicMock spec='list' id='2868486557120'>
>>> mm.foo
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "c:\Python\Python38\lib\unittest\mock.py", line 635, in __getattr__
raise AttributeError("Mock object has no attribute %r" % name)
AttributeError: Mock object has no attribute 'foo'
>>> mm.append
<MagicMock name='mock.append' id='2868486430240'>
>>> mm.append.foo
<MagicMock name='mock.append.foo' id='2868486451408'>
在第二种情况下,被调用的对象也被指定(懒惰地):
>>> ca = mock.create_autospec(list)
>>> ca
<MagicMock spec='list' id='2868486254848'>
>>> ca.foo
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "c:\Python\Python38\lib\unittest\mock.py", line 635, in __getattr__
raise AttributeError("Mock object has no attribute %r" % name)
AttributeError: Mock object has no attribute 'foo'
>>> ca.append
<MagicMock name='mock.append' spec='method_descriptor' id='2868486256336'>
>>> ca.append.foo
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "c:\Python\Python38\lib\unittest\mock.py", line 635, in __getattr__
raise AttributeError("Mock object has no attribute %r" % name)
AttributeError: Mock object has no attribute 'foo'
您的示例代码中显示了一个警告。如果您使用 create_autospec
如此处所示,它的行为就好像该对象是一个 class,而不是一个实例,因此您可以调用它(创建一个实例):
>>> ca = mock.create_autospec(list)
>>> ca()
<NonCallableMagicMock name='mock()' spec='list' id='2868485877280'>
如果你想让它像一个实例一样运行,你必须使用 instance=True
:
>>> ca = mock.create_autospec(list, instance=True)
>>> ca
<NonCallableMagicMock spec='list' id='2868485875024'>
>>> ca()
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
TypeError: 'NonCallableMagicMock' object is not callable
请注意,将 mock.patch
与 autospec=True
结合使用会创建一个模拟,其行为类似于使用 mock.create_autospec
创建的模拟,如 documentation.
中所述
另请注意,调用的 return 值 始终是 MagicMock
,无论实际调用的 return 值如何.因此,即使函数 returns None
,如 list.append
,如果从 mock 调用方法,mock 也会被 returned,而不管规范如何。
我正在尝试了解这两种模拟结构之间的区别以及何时适合使用它们。我在解释器中测试过,例如:
>>> mm = mock.MagicMock(spec=list)
>>> ca = mock.create_autospec(list)
>>> mm
<MagicMock spec='list' id='140372375801232'>
>>> mm()
<MagicMock name='mock()' id='140372384057808'>
>>> mm.append()
<MagicMock name='mock.append()' id='140372375724720'>
>>> mm().append()
<MagicMock name='mock().append()' id='140372375753104'>
>>> ca
<MagicMock spec='list' id='140372384059248'>
>>> ca()
<NonCallableMagicMock name='mock()' spec='list' id='140372384057040'>
>>> ca.append()
<MagicMock name='mock.append()' id='140372375719744'>
>>> ca().append()
<MagicMock name='mock().append()' id='140372375796848'>
>>>
但我不明白为什么“构建”使用 create_autospec
创建的模拟给我 NonCallableMagicMock
而 MagicMock
给了我更多 MagicMock
。 documentation 没有多大帮助。
使用 spec
参数和使用 create_autospec
的主要区别在于递归性。第一种情况,对象本身是指定的,而被调用的对象不是:
>>> mm = mock.MagicMock(spec=list)
>>> mm
<MagicMock spec='list' id='2868486557120'>
>>> mm.foo
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "c:\Python\Python38\lib\unittest\mock.py", line 635, in __getattr__
raise AttributeError("Mock object has no attribute %r" % name)
AttributeError: Mock object has no attribute 'foo'
>>> mm.append
<MagicMock name='mock.append' id='2868486430240'>
>>> mm.append.foo
<MagicMock name='mock.append.foo' id='2868486451408'>
在第二种情况下,被调用的对象也被指定(懒惰地):
>>> ca = mock.create_autospec(list)
>>> ca
<MagicMock spec='list' id='2868486254848'>
>>> ca.foo
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "c:\Python\Python38\lib\unittest\mock.py", line 635, in __getattr__
raise AttributeError("Mock object has no attribute %r" % name)
AttributeError: Mock object has no attribute 'foo'
>>> ca.append
<MagicMock name='mock.append' spec='method_descriptor' id='2868486256336'>
>>> ca.append.foo
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "c:\Python\Python38\lib\unittest\mock.py", line 635, in __getattr__
raise AttributeError("Mock object has no attribute %r" % name)
AttributeError: Mock object has no attribute 'foo'
您的示例代码中显示了一个警告。如果您使用 create_autospec
如此处所示,它的行为就好像该对象是一个 class,而不是一个实例,因此您可以调用它(创建一个实例):
>>> ca = mock.create_autospec(list)
>>> ca()
<NonCallableMagicMock name='mock()' spec='list' id='2868485877280'>
如果你想让它像一个实例一样运行,你必须使用 instance=True
:
>>> ca = mock.create_autospec(list, instance=True)
>>> ca
<NonCallableMagicMock spec='list' id='2868485875024'>
>>> ca()
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
TypeError: 'NonCallableMagicMock' object is not callable
请注意,将 mock.patch
与 autospec=True
结合使用会创建一个模拟,其行为类似于使用 mock.create_autospec
创建的模拟,如 documentation.
另请注意,调用的 return 值 始终是 MagicMock
,无论实际调用的 return 值如何.因此,即使函数 returns None
,如 list.append
,如果从 mock 调用方法,mock 也会被 returned,而不管规范如何。