NullPointerException при вызове смоделированного метода

1

Я пытаюсь высмеять окончательный метод (readChar класса DataInputStream):

MyClassTest

@RunWith(PowerMockRunner.class)
@PrepareForTest(DataInputStream.class)
public class MyClassTest {

    @Test
    public void testMyMethod() throws IOException {
        DataInputStream mockStream = PowerMockito.mock(DataInputStream.class);
        Mockito.when(mockStream.readChar()).thenReturn('a');
        System.out.println(mockStream.readChar());  // OK (print 'a')
        Assert.assertEquals('a', MyClass.myMethod(mockStream));
    }
}

Мои занятия

public class MyClass {
    public static char myMethod(DataInputStream dis) throws IOException {
        return dis.readChar();  // NPE raises
    }
}

Он работает при вызове издевающегося метода в testMyMethod(), но в myMethod() NullPointerException возникает, почему?

РЕДАКТИРОВАТЬ :

Полная ошибка сбоя:

java.lang.NullPointerException
    at java.io.DataInputStream.readChar(Unknown Source)
    at test.test.MyClass.myMethod(MyClass.java:8)
    at test.test.MyClassTest.testMyMethod(MyClassTest.java:23)
    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.internal.runners.TestMethod.invoke(TestMethod.java:59)
    at org.powermock.modules.junit4.internal.impl.PowerMockJUnit44RunnerDelegateImpl$PowerMockJUnit44MethodRunner.runTestMethod(PowerMockJUnit44RunnerDelegateImpl.java:310)
    at org.junit.internal.runners.MethodRoadie$2.run(MethodRoadie.java:79)
    at org.junit.internal.runners.MethodRoadie.runBeforesThenTestThenAfters(MethodRoadie.java:87)
    at org.powermock.modules.junit4.internal.impl.PowerMockJUnit44RunnerDelegateImpl$PowerMockJUnit44MethodRunner.executeTest(PowerMockJUnit44RunnerDelegateImpl.java:294)
    at org.powermock.modules.junit4.internal.impl.PowerMockJUnit44RunnerDelegateImpl$PowerMockJUnit44MethodRunner.runBeforesThenTestThenAfters(PowerMockJUnit44RunnerDelegateImpl.java:282)
    at org.junit.internal.runners.MethodRoadie.runTest(MethodRoadie.java:77)
    at org.junit.internal.runners.MethodRoadie.run(MethodRoadie.java:42)
    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:27)
    at org.junit.internal.runners.ClassRoadie.runProtected(ClassRoadie.java:37)
    at org.powermock.modules.junit4.internal.impl.PowerMockJUnit44RunnerDelegateImpl.run(PowerMockJUnit44RunnerDelegateImpl.java:122)
    at org.powermock.modules.junit4.common.internal.impl.JUnit4TestSuiteChunkerImpl.run(JUnit4TestSuiteChunkerImpl.java:104)
    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:50)
    at org.eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.java:38)
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:459)
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:675)
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.run(RemoteTestRunner.java:382)
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.main(RemoteTestRunner.java:192)
Теги:
powermock
mockito

2 ответа

1
Лучший ответ

DataInputStream - это "системный" класс из JVM, который, вероятно, уже загружен JVM. @PrepareForTest должен будет удалить final модификатор из методов (чтобы быть в состоянии издеваться), но он не может сделать этого для уже загруженных классов (HotSpot JVM не поддерживает изменения подписи класса для уже загруженных классов), и это вероятно, почему вы получаете это исключение.

К счастью, там также интерфейс DataInput реализованный DataInputStream - возможно, вы можете попробовать насмехаться не с DataInputStream а с DataInput, для этого вам даже не нужен PowerMock, просто Mockito.

  • 0
    Я высмеял DataInput вместо DataInputStream, и он работает. Спасибо за ваше четкое объяснение и вашу помощь!
2

Сначала этот код является мафиозным антипаттерном: не издевайтесь над типами, которыми вы не владеете! (см. этот qaru.site/questions/1018172/...)

Во-вторых, DataInputStream - это класс JDK PowerMock, который не может использовать тот же хакерский загрузчик классов, который модифицирует байтовый код.

Для этого есть решение и два возможных трюка:

  • использовать интерфейс
  • инкапсулируйте вызов в классе, который у вас есть, затем подготовьте этот класс, задокументированный здесь, в коде google, или даже лучше сделайте свой собственный макетный класс (без необходимости powermock)
  • использовать агента, задокументированного здесь, в коде google

Первый вариант, безусловно, самый лучший, и первые два варианта также позволяют избежать этого ложного антипаттера.

Ещё вопросы

Сообщество Overcoder
Наверх
Меню