Java 在测试方法内模拟logger.debug()调用

Java 在测试方法内模拟logger.debug()调用,java,mockito,spy,Java,Mockito,Spy,我有一个我正在尝试测试的方法,比如methodToTest()。在这个方法中,调用了一个我认为是Logger变量的函数,比如: logger.debug("this is a logger debug example"); 问题是logger变量是在类之外声明的(它似乎是继承的),我没有该类的源代码 当我创建测试类的实例,然后调用methodToTest()方法时,我在logger.debug语句中得到一个NullPointerException 我怎样才能完全模仿记录器?它在测试期间没有任何

我有一个我正在尝试测试的方法,比如
methodToTest()
。在这个方法中,调用了一个我认为是
Logger
变量的函数,比如:

logger.debug("this is a logger debug example");
问题是logger变量是在类之外声明的(它似乎是继承的),我没有该类的源代码

当我创建测试类的实例,然后调用
methodToTest()
方法时,我在
logger.debug
语句中得到一个
NullPointerException

我怎样才能完全模仿记录器?它在测试期间没有任何用处

when(logger.debug("")).doNothing();
即使在测试类中创建
Logger
变量时,也会给我一个错误

我想我可能应该使用间谍,但我不知道如何使用,也不知道如何使用我当前的
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)

假设您可以更改代码,我认为最简单/最干净的方法是将日志接口注入到您正在测试的类中。然后,对
.debug
方法进行模拟或使用无操作的假日志记录实现。

您可以使用反射来实现:

Field f = Yourclass.class.getDeclaredField("logger");
f.setAccessible(true);
f.set(Yourclass.class, yourMock );
当您需要搜索字段直至超类时,您可以进行一些查找循环:

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;
    }
}

但是这不是假设
Logger
变量是在我的测试类中声明的吗?我上面提到的不是。你可以为你的领域签出超类,我会更新那个不起作用的答案。声明
logger
变量的不是直接超类,我也不确定与该类的关系,因为我没有声明该logger的类的源代码。实际上,如果(f!=null)
代码行允许我的测试通过,则删除了
。我不确定我是否理解这一点。显然,要访问最终字段,您需要再做一些工作(hack!~1)。我会再次更新我的答案。此外,我犯了一个错误,建议您使用错误的
field.set
方法。