java - 在测试方法中模拟 logger.debug() 调用

标签 java mockito spy

我有一个正在尝试测试的方法,例如 methodToTest() 。这个方法内部是对我认为是 Logger 的调用。变量,例如:

logger.debug("this is a logger debug example");

问题是记录器变量是在类外部声明的(它似乎是继承的),我没有源代码。

当我创建测试类的实例,然后调用 methodToTest()方法,我得到一个 NullPointerExceptionlogger.debug声明。

我怎样才能完全模拟记录器?它在测试过程中没有任何用处。

when(logger.debug("")).doNothing();

给我一​​个错误,即使我创建了 Logger测试类中的变量。

我想我也许应该使用 spy Activity ,但我不确定如何使用,然后不确定如何处理我当前的 when().thenReturn();声明。

根据 @kamil 的建议,

f.set( LM2DImportMultiTaskWorker.class, null );

代码抛出以下错误:

java.lang.IllegalArgumentException: Can not set org.apache.log4j.Logger field com.navteq.cf.foundation.process.OperationSkeleton.logger to java.lang.Class
    at sun.reflect.UnsafeFieldAccessorImpl.throwSetIllegalArgumentException(Unknown Source)
    at sun.reflect.UnsafeFieldAccessorImpl.throwSetIllegalArgumentException(Unknown Source)
    at sun.reflect.UnsafeFieldAccessorImpl.ensureObj(Unknown Source)
    at sun.reflect.UnsafeObjectFieldAccessorImpl.set(Unknown Source)
    at java.lang.reflect.Field.set(Unknown Source)
    at com.navteq.rdf.base.task.LM2DImportMultiTaskWorkerTest.testGetCountFromDB(LM2DImportMultiTaskWorkerTest.java:101)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(Unknown Source)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(Unknown Source)
    at java.lang.reflect.Method.invoke(Unknown Source)
    at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:44)
    at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:15)
    at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:41)
    at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:20)
    at org.junit.internal.runners.statements.RunBefores.evaluate(RunBefores.java:28)
    at org.junit.internal.runners.statements.RunAfters.evaluate(RunAfters.java:31)
    at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:73)
    at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:46)
    at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:180)
    at org.junit.runners.ParentRunner.access$000(ParentRunner.java:41)
    at org.junit.runners.ParentRunner$1.evaluate(ParentRunner.java:173)
    at org.junit.internal.runners.statements.RunBefores.evaluate(RunBefores.java:28)
    at org.junit.internal.runners.statements.RunAfters.evaluate(RunAfters.java:31)
    at org.junit.runners.ParentRunner.run(ParentRunner.java:220)
    at org.eclipse.jdt.internal.junit4.runner.JUnit4TestReference.run(JUnit4TestReference.java:50)
    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)

最佳答案

您可以像这样使用反射:

Field f = Yourclass.class.getDeclaredField("logger");
f.setAccessible(true);
f.set(Yourclass.class, yourMock );

当您需要搜索您的字段直至父类(super class)时,您可以进行一些查找循环:

Class clazz = Yourclass.class;

while ( clazz != null ) {
    Field f = null;
        try {
            f = clazz.getDeclaredField( "logger" );
        } catch ( NoSuchFieldException e ) {
            clazz = clazz.getSuperclass();
        }
    if ( f != null ) {
        f.setAccessible( true );

        //here we must hack final modifier for field
        Field modifiersField = Field.class.getDeclaredField("modifiers");
        modifiersField.setAccessible(true);
        modifiersField.setInt(f, f.getModifiers() & ~Modifier.FINAL);

        //here pass your object which should contain logger field
        f.set( yourObject,  mock(mockSomeClass) );
        break;
    }
}

关于java - 在测试方法中模拟 logger.debug() 调用,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/24494550/

相关文章:

java - 如何在要测试的类中模拟抽象类方法

java - 获取 ActiveMQ 队列长度的任何简单方法?

java - JFrame 中的更新信息

java - 从回调运行计时器时不调用 Looper.prepare()

java - 使用 Apache Common Net 从 FTP 下载到设备的损坏的 PDF 文件

Java模拟函数来改变它的行为

java - 如何验证使用 "null"参数调用 spy

java - 在 gradle 项目中使用 Junit 和 Mockito 动态测试文件(xml、json、csv、txt)中的测试数据

java - 无法从数据库获取数据-mockito

java - 为什么我在模拟时会收到 NullPointerException?