【问题标题】:How to mock a non-static private inner java class with PowerMock and EasyMock?如何使用 PowerMock 和 EasyMock 模拟非静态私有内部 Java 类?
【发布时间】:2013-11-10 15:45:59
【问题描述】:

这个StackOverflowAnswer 对我不起作用,所以我重新创建了这个问题。

如何通过仅修改 testClass (EnclosureClassTest.java) 而不是实现 ( EnclosureClass.javaInnerClassType.java)?

要求:

  • 使用 PowerMock 和 EasyMock 模拟非静态私有内部 Java 类。

EnclosureClassTest.java:

import java.lang.reflect.Constructor;

import org.junit.Test;
import org.junit.runner.RunWith;
import org.powermock.api.easymock.PowerMock;
import org.powermock.core.classloader.annotations.PrepareForTest;
import org.powermock.modules.junit4.PowerMockRunner;
import org.powermock.reflect.Whitebox;

@RunWith(PowerMockRunner.class)
@PrepareForTest({ EnclosingClass.class })
public class EnclosingClassTest {

    EnclosingClass cut;

    @Test
    public void testMethod() throws Exception {
        /* Initialization */

        /* Mock Setup */
        Class clazz = Whitebox.getInnerClassType(EnclosingClass.class,
                "InnerClass");
        Constructor constructor = Whitebox.getConstructor(clazz, null);
        InnerClassType innerClass = (InnerClassType) constructor
                .newInstance(null);

        /* Activate the Mocks */
        PowerMock.replayAll();

        /* Test Method */

        /* Asserts */
        PowerMock.verifyAll();

    }
}

EnclosureClass.java:

public class EnclosingClass {

    private class InnerClass implements InnerClassType {
        public InnerClass() { /* do stuff */
        }

        public int innerClassMethod() {
            return 0;
        }
    }
}

InnerClassType.java:

public interface InnerClassType {
    public int innerClassMethod();
}

EnclosureClassTest.testMethod() 抛出 异常:

org.powermock.reflect.exceptions.ConstructorNotFoundException: Failed to lookup constructor with parameter types [ <none> ] in class EnclosingClass$InnerClass.
    at org.powermock.reflect.internal.WhiteboxImpl.getConstructor(WhiteboxImpl.java:258)
    at org.powermock.reflect.Whitebox.getConstructor(Whitebox.java:158)
    at EnclosingClassTest.testMethod(EnclosingClassTest.java:25)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:60)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:37)
    at java.lang.reflect.Method.invoke(Method.java:611)
    at org.junit.internal.runners.TestMethod.invoke(TestMethod.java:66)
    at org.powermock.modules.junit4.internal.impl.PowerMockJUnit44RunnerDelegateImpl$PowerMockJUnit44MethodRunner.runTestMethod(PowerMockJUnit44RunnerDelegateImpl.java:310)
    at org.junit.internal.runners.MethodRoadie$2.run(MethodRoadie.java:86)
    at org.junit.internal.runners.MethodRoadie.runBeforesThenTestThenAfters(MethodRoadie.java:94)
    at org.powermock.modules.junit4.internal.impl.PowerMockJUnit44RunnerDelegateImpl$PowerMockJUnit44MethodRunner.executeTest(PowerMockJUnit44RunnerDelegateImpl.java:294)
    at org.powermock.modules.junit4.internal.impl.PowerMockJUnit47RunnerDelegateImpl$PowerMockJUnit47MethodRunner.executeTestInSuper(PowerMockJUnit47RunnerDelegateImpl.java:127)
    at org.powermock.modules.junit4.internal.impl.PowerMockJUnit47RunnerDelegateImpl$PowerMockJUnit47MethodRunner.executeTest(PowerMockJUnit47RunnerDelegateImpl.java:82)
    at org.powermock.modules.junit4.internal.impl.PowerMockJUnit44RunnerDelegateImpl$PowerMockJUnit44MethodRunner.runBeforesThenTestThenAfters(PowerMockJUnit44RunnerDelegateImpl.java:282)
    at org.junit.internal.runners.MethodRoadie.runTest(MethodRoadie.java:84)
    at org.junit.internal.runners.MethodRoadie.run(MethodRoadie.java:49)
    at org.powermock.modules.junit4.internal.impl.PowerMockJUnit44RunnerDelegateImpl.invokeTestMethod(PowerMockJUnit44RunnerDelegateImpl.java:207)
    at org.powermock.modules.junit4.internal.impl.PowerMockJUnit44RunnerDelegateImpl.runMethods(PowerMockJUnit44RunnerDelegateImpl.java:146)
    at org.powermock.modules.junit4.internal.impl.PowerMockJUnit44RunnerDelegateImpl$1.run(PowerMockJUnit44RunnerDelegateImpl.java:120)
    at org.junit.internal.runners.ClassRoadie.runUnprotected(ClassRoadie.java:34)
    at org.junit.internal.runners.ClassRoadie.runProtected(ClassRoadie.java:44)
    at org.powermock.modules.junit4.internal.impl.PowerMockJUnit44RunnerDelegateImpl.run(PowerMockJUnit44RunnerDelegateImpl.java:118)
    at org.powermock.modules.junit4.common.internal.impl.JUnit4TestSuiteChunkerImpl.run(JUnit4TestSuiteChunkerImpl.java:101)
    at org.powermock.modules.junit4.common.internal.impl.AbstractCommonPowerMockRunner.run(AbstractCommonPowerMockRunner.java:53)
    at org.powermock.modules.junit4.PowerMockRunner.run(PowerMockRunner.java:53)
    at org.eclipse.jdt.internal.junit4.runner.JUnit4TestReference.run(JUnit4TestReference.java:49)
    at org.eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.java:38)
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:467)
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:683)
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.run(RemoteTestRunner.java:390)
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.main(RemoteTestRunner.java:197)
Caused by: java.lang.NoSuchMethodException: EnclosingClass$InnerClass.<init>()
    at java.lang.Class.throwNoSuchMethodException(Class.java:286)
    at java.lang.Class.getDeclaredConstructor(Class.java:429)
    at org.powermock.reflect.internal.WhiteboxImpl.getConstructor(WhiteboxImpl.java:250)
    ... 31 more

【问题讨论】:

    标签: java junit mocking easymock powermock


    【解决方案1】:

    我认为你需要这样定义类:

    public class EnclosingClass {
    
        private static class InnerClass implements InnerClassType {
            public InnerClass() { /* do stuff */
            }
    
            public int innerClassMethod() {
                return 0;
            }
        }
    }
    

    在您的定义中,InnerClass 应该低于 EnclosureClass 的一个实例。当你不使用反射时,如果你想实例化一个 InnerClass,请尝试一下会发生什么。

    可以直接获取构造函数。并像这样传入一个 EnclosureClass 的实例:

    public void testMethod() throws Exception {
            /* Initialization */
    
            /* Mock Setup */
            Class clazz = Whitebox.getInnerClassType(EnclosingClass.class,
                    "InnerClass");
    
            Constructor[] constructors = clazz.getConstructors();
            Constructor c  = constructors[0];
    //        c.setAccessible(true);
            InnerClassType innerClass = (InnerClassType) c
                    .newInstance(new EnclosingClass());
    
            /* Activate the Mocks */
            PowerMock.replayAll();
    
            /* Test Method */
    
            /* Asserts */
            PowerMock.verifyAll();
    
        }
    

    【讨论】:

    • 如果我使用您的“public static class EnclosureClass”建议,那么我会收到此错误:“EnclosingClass 类的非法修饰符;只有 public、abstract 和 final 是允许”@Terry Zhao
    • 但这就是模拟静态私有内部类的方法。 @特里赵
    • 该问题要求对 EnclosureClass.java 和 InnerClassType.java 的实现不进行任何更改。实施是一成不变的。 @特里赵
    • 我运行了您(@Terry Zhao)更新(修改)的 testMethod(),它挂起了我的 Eclipse。如果我在这一行 Class clazz = Whitebox.getInnerClassType(EnclosureClass.class, "InnerClass"); 上放置一个断点,它甚至不会命中这一行。我还尝试将 InnerClassType.class 添加到 @PrepareForTest 注释参数中。
    • 我不确定你的代码或环境发生了什么,在我用 Eclipse 中定义的替换从 powermock 下载的 Junit lib 后,我的 Eclipse 为这个测试用例亮了绿灯。跨度>
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 2017-05-19
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2012-03-30
    • 2012-02-09
    • 2016-12-05
    相关资源
    最近更新 更多