androidx.lifecycle:lifecycle-*:2.2.0升级带来的问题

将lifecycle-*相关依赖升级到2.2.0时,项目中的ViewModelProviders显示被弃用

查看官方文档发现:


ViewModelProviders已被弃用

需要使用 ViewModelProvider(ViewModelStoreOwner) 方法代替
这里的ViewModelStoreOwner是一个接口,Fragment和Activity都实现了该接口,所以直接传以前的参数就可以
全局替换(ctrl+shift+r)依赖和改行代码,完事

求豆麻袋!事情并没有这么简单

应用启动就会崩溃,报如下错误:

java.lang.RuntimeException:Cannot create an instance of class **ViewModel
  
cause:java.lang.Class<*ViewModel> has no zero argument constructor

无法创建ViewModel,提示ViewModel没有无参构造函数

撸一把源码看看到底哪里出了问题,官方总不能给个没法用的东西

在Android中使用ViewModel一般是通过继承AndroidViewModel进行使用,而AndroidViewModel只有一个构造方法:

public AndroidViewModel(@NonNull Application application) {
    mApplication = application;
}

而其父类ViewModel类本身是没有构造方法的,即默认实现了无参构造方法,而AndroidViewModel则不存在无参构造方法

所以第一个解决办法就是直接使用ViewModel来代替AndroidViewModel(没有具体试过,应该是可以的)

但是使用AndroidViewModel肯定也是有相应的解决办法的,那就需要继续看ViewModelProvider的源码,在创建ViewModel时调用了以下的方法:

public ViewModelProvider(@NonNull ViewModelStoreOwner owner) {
    //这里调用到了另一个构造方法,第一个参数为Fragment或者Activity;
    //对于第二个参数来说,Fragemnt和Activtiy没有实现HasDefaultViewModelProviderFactory这个接口,所以对于这种情况来说,就是调用了 NewInstanceFactory.getInstance()来生成一个工厂
    this(owner.getViewModelStore(), owner instanceof HasDefaultViewModelProviderFactory
            ? ((HasDefaultViewModelProviderFactory) owner).getDefaultViewModelProviderFactory()
            : NewInstanceFactory.getInstance());
}

继续往下看,NewInstanceFactory类中出现了这样一行注释:

/**
 * Simple factory, which calls empty constructor on the give class.
 */

在该工厂类的create方法中也直接调用了

return modelClass.newInstance();

ok,既然找到了问题,就肯定要找到解决的办法,贴心的官方在该类下方直接就跟着AndroidViewModelFactory类,继承自该类,其中比该类多了application相关的操作,并且调用了以Application为单独参数的构造方法

那么完美解决办法就是在使用ViewModelProvider时多传一个参数,即AndroidViewModelFactory的实例,完整代码如下:

mViewModel= ViewModelProvider(this,ViewModelProvider.AndroidViewModelFactory.getInstance(application))[XXXViewModel::class.java]

你可能感兴趣的:(androidx.lifecycle:lifecycle-*:2.2.0升级带来的问题)