PowerMock mockStatic 没有捕获模拟的 static void 方法调用

PowerMock mockStatic didn't catch mocked static void method call

我尝试在 Mockito 上使用 PowerMock 模拟静态 void 方法,但效果不佳。

我的示例代码:

BlackTempleTest.java

package com;

import org.junit.Assert;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.mockito.invocation.InvocationOnMock;
import org.mockito.stubbing.Answer;
import org.powermock.api.mockito.PowerMockito;
import org.powermock.core.classloader.annotations.PrepareForTest;
import org.powermock.modules.junit4.PowerMockRunner;

import com.BlackTempleTest.Illidan;

@RunWith(PowerMockRunner.class)
@PrepareForTest({ Illidan.class, EvilBrother.class })
public class BlackTempleTest {

    Answer<Object> defaultAnswer = new Answer<Object>() {
        @Override
        public Object answer(InvocationOnMock invocation) throws Throwable {
            System.out.println("Haha!");
            return null;
        }
    };

    @Test(expected = AssertionError.class)
    public void testIllidanFight() throws Exception {
        Illidan.startFight();
    }

    @Test
    public void testCheatOnIllidanFight() throws Exception {
        PowerMockito.mockStatic(Illidan.class, defaultAnswer);

        Illidan.startFight();
    }

    @Test(expected = AssertionError.class)
    public void testEvilBrotherFight() throws Exception {
        EvilBrother.startFight();
    }

    // dont work
    @Test
    public void testCheatOnEvilBrotherFight() throws Exception {
        PowerMockito.mockStatic(EvilBrother.class, defaultAnswer);

        EvilBrother.startFight();
    }

    static class Illidan {
        static void startFight() {
            Assert.fail("You are not prepared!");
        }
    }
}

EvilBrother.java

package com;

import com.BlackTempleTest.Illidan;

public class EvilBrother extends Illidan {

}

我的问题是,嵌套的 class 被 @PrepareForTest 和 PowerMockito.mockStatic 的组合按预期模拟,但是如果 class 在它自己的 class 文件,这些语句不起作用。

如何修复这个测试?

编辑:

    PowerMockito.doAnswer(defaultAnswer).when(EvilBrother.class, "startFight");

可以通过 Powermock 传递错误的调用,但是 Assert.fail 会被执行。

java.lang.AssertionError: You are not prepared!
    at org.junit.Assert.fail(Assert.java:88)
    at com.BlackTempleTest$Illidan.startFight(BlackTempleTest.java:54)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:606)
    at org.powermock.reflect.internal.WhiteboxImpl.performMethodInvocation(WhiteboxImpl.java:1873)
    at org.powermock.reflect.internal.WhiteboxImpl.doInvokeMethod(WhiteboxImpl.java:773)
    at org.powermock.reflect.internal.WhiteboxImpl.invokeMethod(WhiteboxImpl.java:753)
    at org.powermock.reflect.Whitebox.invokeMethod(Whitebox.java:466)
    at org.powermock.api.mockito.internal.expectation.PowerMockitoStubberImpl.when(PowerMockitoStubberImpl.java:106)
    at com.BlackTempleTest.testCheatOnEvilBrotherFight(BlackTempleTest.java:47)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)

解决方案是,你必须模拟伊利丹而不是 EvilBrother,即使你调用 EvilBrother.startFight 因为方法是继承的。

@Test
public void testCheatOnEvilBrotherFight() throws Exception {
    PowerMockito.mockStatic(Illidan.class, defaultAnswer);

    EvilBrother.startFight();
}