java - 进行单元测试时无法将Value 设置为 MutableLiveData - 抛出 java.lang.reflect.InvocableTargetException

标签 java android unit-testing android-livedata android-viewmodel

我有一个 User 类,它有一个构建器,并且该类的构造函数是私有(private)的。

public class User {

    private String name;

    private User() {
    }

    public static class UserBuilder {
        private User user;

        public UserBuilder() {
            user = new User();
        }

        public UserBuilder withName(String name) {
            user.name = name;
            return this;
        }

        public User build() {
            return user;
        }
    }
}

然后我想要一个 ViewModel 类,我希望有一个 MutableLiveData 用于从我的 Activity 中观察到的 User 类/分段。

public class UserViewModel extends ViewModel {

    private User.UserBuilder userBuilder;
    private MutableLiveData<User> user;

    public UserViewModel() {
        userBuilder = new User.UserBuilder();
        user = new MutableLiveData<>(userBuilder.build());
    }

    public void setName(String name) {
        userBuilder.withName(name);
        user.setValue(userBuilder.build());
    }

    public MutableLiveData<User> getUser() {
        user.setValue(userBuilder.build());
        return user;
    }
}

在上面的实现中,我发现在从我的单元测试中调用 setValue 函数时,它抛出一个 NullPointerException ,后跟一个 InitationTargetException

我搜索了原因,看起来,User 类需要有一个 public 构造函数才能与 MutableLiveData 一起使用。然而,事实并非如此,我尝试创建一个 public 构造函数,但它仍然失败。

我正在尝试编写一些单元测试,但在 logcat 中收到以下错误。请注意,我正在使用以下内容来解决我认为的 mainThread 错误。

@Rule
public InstantTaskExecutorRule instantTaskExecutorRule = new InstantTaskExecutorRule();

这是我在单元测试中尝试做的事情。

@RunWith(MockitoJUnitRunner.class)
public class UserViewModelTest {

    private UserViewModel userViewModel;

    @Mock
    private Observer<User> userObserver;

    @Rule
    public InstantTaskExecutorRule instantTaskExecutorRule = new InstantTaskExecutorRule();

    @Before
    public void setup() {
        initMocks(this);
        userViewModel = new userViewModel();
        userViewModel.getUser().observeForever(userObserver);
    }
}

异常(exception)情况在以下行中。

user.setValue(userBuilder.build());

这是日志猫。

java.lang.NullPointerException
    at androidx.arch.core.executor.DefaultTaskExecutor.isMainThread(DefaultTaskExecutor.java:77)
    at androidx.arch.core.executor.ArchTaskExecutor.isMainThread(ArchTaskExecutor.java:116)
    at androidx.lifecycle.LiveData.assertMainThread(LiveData.java:460)
    at androidx.lifecycle.LiveData.setValue(LiveData.java:304)
    at androidx.lifecycle.MutableLiveData.setValue(MutableLiveData.java:50)
    at com.example.viewmodel.UserViewModel.getUser(UserViewModel.java:79)
    at com.example.viewmodels.UserViewModelTest.setup(UserViewModelTest.java:66)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:498)
    at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:50)
    at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12)
    at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:47)
    at org.junit.internal.runners.statements.RunBefores.evaluate(RunBefores.java:24)
    at org.mockito.internal.runners.DefaultInternalRunner$1$1.evaluate(DefaultInternalRunner.java:44)
    at org.junit.rules.TestWatcher$1.evaluate(TestWatcher.java:55)
    at org.junit.rules.RunRules.evaluate(RunRules.java:20)
    at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:325)
    at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:78)
    at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:57)
    at org.junit.runners.ParentRunner$3.run(ParentRunner.java:290)
    at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:71)
    at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:288)
    at org.junit.runners.ParentRunner.access$000(ParentRunner.java:58)
    at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:268)
    at org.junit.runners.ParentRunner.run(ParentRunner.java:363)
    at org.mockito.internal.runners.DefaultInternalRunner$1.run(DefaultInternalRunner.java:74)
    at org.mockito.internal.runners.DefaultInternalRunner.run(DefaultInternalRunner.java:80)
    at org.mockito.internal.runners.StrictRunner.run(StrictRunner.java:39)
    at org.mockito.junit.MockitoJUnitRunner.run(MockitoJUnitRunner.java:163)
    at org.junit.runners.Suite.runChild(Suite.java:128)
    at org.junit.runners.Suite.runChild(Suite.java:27)
    at org.junit.runners.ParentRunner$3.run(ParentRunner.java:290)
    at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:71)
    at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:288)
    at org.junit.runners.ParentRunner.access$000(ParentRunner.java:58)
    at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:268)
    at org.junit.runners.ParentRunner.run(ParentRunner.java:363)
    at org.junit.runner.JUnitCore.run(JUnitCore.java:137)
    at com.intellij.junit4.JUnit4IdeaTestRunner.startRunnerWithArgs(JUnit4IdeaTestRunner.java:68)
    at com.intellij.rt.execution.junit.IdeaTestRunner$Repeater.startRunnerWithArgs(IdeaTestRunner.java:47)
    at com.intellij.rt.execution.junit.JUnitStarter.prepareStreamsAndStart(JUnitStarter.java:242)
    at com.intellij.rt.execution.junit.JUnitStarter.main(JUnitStarter.java:70)

提前致谢!

最佳答案

终于,我明白了是什么原因导致了异常。看起来当 Observer 设置为 ViewModel 时,需要一些时间来初始化,同时,如果我们尝试 setValue a LiveData,它会抛出 NullPointerException,因为 ViewModel 尚未初始化。这是测试环境设置的问题,正如 @CommonsWare 在本问题前面的评论中所建议的那样。我真的很感激,因为它引导我找到了摆脱异常的方法。

我决定在 Kotlin 中编写单元测试,因为使用 lateinit 变量更容易。我发现了一个很好的tutorial here 。我在此处发布与测试 ViewModel 相关的代码。

首先创建一个名为MockUtils.kt的辅助类。

import org.mockito.Mockito

/**
 * Helper function to mock classes with types (generics)
 */
inline fun <reified T> mock(): T = Mockito.mock(T::class.java)

最后,测试类应如下所示。

class UserViewModelTest {

    @get:Rule
    val rule = InstantTaskExecutorRule()

    private lateinit var viewModel: UserViewModel

    private val observer: Observer<User> = mock()

    @Before
    fun before() {
        viewModel = UserViewModel()
        viewModel.user.observeForever(observer)
    }

    @Test
    fun testUserViewModel() {
        val expectedUser = viewModel.user.value
        viewModel.setName("John")

        val captor = ArgumentCaptor.forClass(User::class.java)
        captor.run {
            verify(observer, times(2)).onChanged(capture())
            assertNotNull(expectedUser)
            assertEquals("John", value.name)
        }
    }
}

为了消除异常,我必须解决另一个问题。我认为如果我在这里提到这一点,会对其他开发人员有所帮助。

我错误地使用了不同版本的 InstantTaskExecutorRuleLiveData。您可以查看tweet from Jeroen Mols here 。我有同样的问题。 LiveData 是从 androidx 包导入的,而 InstantTaskExecutorRule 是使用 android 包导入的 - 必须修复这个问题如下。

implementation "androidx.lifecycle:lifecycle-viewmodel:2.1.0"
implementation "androidx.lifecycle:lifecycle-livedata:2.1.0"
implementation "androidx.lifecycle:lifecycle-extensions:2.1.0"
testImplementation "androidx.arch.core:core-testing:2.1.0"

我创建了一个Github repository使用此问题中的简单类,以便任何人都可以使用自己的开发环境检查和运行单元测试。快乐编码!

关于java - 进行单元测试时无法将Value 设置为 MutableLiveData - 抛出 java.lang.reflect.InvocableTargetException,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58615467/

相关文章:

angular - 升级到 Karma 2.0.0 后 ng 测试不起作用

javascript - 如何使用 Angular 8 单元测试检查图像是否存在于给定路径中?

java - 尝试使用 Castor 紧凑地序列化对象

java - 如何在java中找到具有相同键的多个记录中的最小值?

android - 这个布局该怎么做呢? (图片上的链接)

android - 从我的应用程序调用另一个应用程序

javascript - 在单元测试中显示对象详细信息

java - 像 Log4j 这样的日志框架如何保证日志语句的顺序?

Java 公共(public)类,具有带有非公共(public)参数的公共(public)构造函数。为什么?

android - 无法弹出导航库的返回堆栈