这是我的班级:
public function __construct(Manager $moduleManager, Source\Yesno $yesNo)
{
$this->moduleManager = $moduleManager;
$this->yesNo = $yesNo;
}
public function my1()
{
$this->moduleManager->isOutputEnabled('');
$this->yesNo->toOptionArray();
}
public function my2()
{
$this->moduleManager->isOutputEnabled('');
$this->yesNo->toOptionArray();
}
这是我的测试:
...
$this->observerMock = $this->getMock(
'path\to\Observer',
null,
[$this->moduleManagerMock, $this->yesNoMock],
'',
true
);
...
public function testMy1()
{
$this->moduleManagerMock->expects($this->exactly(2))->method('isOutputEnabled');
$this->yesNoMock->expects($this->exactly(2))->method('toOptionsArray');
$this->observerMock->my1();
$this->observerMock->my2();
}
测试返回:
方法名称的期望失败等于 当被调用2次时。预计方法被称为2次, 实际上叫了0次。
我的问题是:我几次面对这样的事情,但每次我都无法理解发生的事情。为什么第一个期望是正确的,第二个不合适?
我忘了说我几次遇到这样的情况。 这是我注意到的。使用xDebug我看到了内部测试
[this]
[moduleManager] => ModuleManager_Mock_Name_<hash #1>
[yesNo] => YesNo_Mock_Name_<hash #2>
[observerObject] =>
[moduleManager] => ModuleManager_Mock_Name_<hash #1>
[yesNo] => YesNo_Mock_Name_<hash #3> <<<<<<<<<<<<<<<<<<<<<<<<<<<<<<
moduleManager对象在unittest对象和内部观察者对象中具有相同的缓存。如果我在moduleMatcher上应用smth - 它会出现在两个地方
$ unittest-&gt; yes的哈希值与$ observerObject哈希值不同。如果我为某些方法设置匹配器 - 它仅适用于unittest类!
为什么会这样?如何防止创建不同的对象
通过对象管理器创建对象时
$this->observerMock = $objectManager->getObject(
'Observer',
[
'moduleManager' => $this->moduleManagerMock,
'yesNo' => $this->yesNoMock,
]
);
变量'moduleManager'和'yesNo'应该与构造函数中的变量相同:
public function __construct(Manager $moduleManager, Source\Yesno $yesNo)
{
$this->moduleManager = $moduleManager;
$this->yesNo = $yesNo;
}
以下是phpunit检查此代码的代码:
foreach ($method->getParameters() as $parameter) {
$parameterName = $parameter->getName();
$argClassName = null;
$defaultValue = null;
if (array_key_exists($parameterName, $arguments)) {
$constructArguments[$parameterName] = $arguments[$parameterName];
continue;
}
答案 0 :(得分:1)
通常情况下,您不应该模拟/存储您正在测试的系统本身。因为在您的测试用例中,$this->observerMock
对象本身就是一个存根对象(它模仿了另一个类的接口,但没有提供任何实现)。
这意味着方法m1
和m2
也是模拟方法,它们在被调用时不会执行任何操作。随后,您的依赖项(moduleManagerMock
和yesNoMock
)上的模拟方法将永远不会被称为,这就是您的期望失败的原因。
要正确测试您想要的行为,请直接使用Observer
课程:
public function setUp() {
$this->moduleManagerMock = $this->getMock(/*...*/);
$this->yesNoMock = $this->getMock(/*...*/);
// Do not generate a mock object of "Observer", but use the class
// under test itself!
$this->observer = new Observer(
$this->moduleManagerMock,
$this->yesNoMock
);
}
public function testM1() {
$this->moduleManagerMock->expects($this->exactly(2))
->method('isOutputEnabled');
$this->yesNoMock->expects($this->exactly(2))
->method('toOptionsArray');
$this->observer->my1();
$this->observer->my2();
}