ChatGPT解决这个技术问题 Extra ChatGPT

查看 ViewModelGoogle docs,他们展示了以下关于如何获取 ViewModel 的示例代码:

val model = ViewModelProviders.of(this).get(MyViewModel::class.java)

当使用最新的依赖项 android.arch.lifecycle:extensions:1.1.1 时,没有这样的类 ViewModelProviders

转到 ViewModelProvidersdocumentation,我看到一条评论说:

此类在 API 级别 1.1.0 中已弃用。使用 ViewModelProvider.AndroidViewModelFactory

问题是,在尝试使用 ViewModelProvider.AndroidViewModelFactory 时,找不到等效的 of 方法来获取 ViewModel 的实例。

我尝试做的事情:

ViewModelProvider.AndroidViewModelFactory.getInstance(application).create(PlayerViewHolder::class.java)

因此方法 create 的名称,我每次调用它时都会得到一个 ViewModel 的新实例,这不是我想要的。

有什么想法可以替换上面不推荐使用的代码吗?

使用单例模式或更改为 AndroidX

H
Haseeb Hassan Asif

我使用生命周期扩展 2.2.0 版本:

implementation "androidx.lifecycle:lifecycle-extensions:2.2.0" 

它应该工作,使用 ViewModelProvider 构造函数。

// With ViewModelFactory   
val viewModel = ViewModelProvider(this, YourViewModelFactory).get(YourViewModel::class.java)


//Without ViewModelFactory
val viewModel = ViewModelProvider(this).get(YourViewModel::class.java)

2020/5/15 更新

我找到了另一种优雅的方式来实现这一点,Android KTX 可以提供帮助

implementation "androidx.fragment:fragment-ktx:1.2.4"
val viewmodel: MYViewModel by viewModels()
val viewmodel: MYViewModel by viewModels { myFactory } //With factory

参考:https://developer.android.com/reference/kotlin/androidx/fragment/app/package-summary#viewmodels

2020/06/25:更正了代表的情况


我有点厌倦了谷歌不断的“折旧”周期。
ViewModelProviders.of() 已被弃用。您可以将 Fragment 或 FragmentActivity 传递给新的 ViewModelProvider(ViewModelStoreOwner) 构造函数以实现相同的功能。 (aosp/1009889)
Java 等效项:YourViewModel viewModel = new ViewModelProvider(this).get(YourViewModel.class);
我担心有一天 Google 会弃用 TextViewEditText 并引入一个新组件。
androidx 生命周期视图模型 2.1.0 没有没有工厂的构造函数。简单地使用(this)不起作用。 ViewModelProvider(ViewModelStoreOwner, Factory) 和 ViewModelProvider(ViewModelStore, Factory) 是唯一的构造函数
A
Alex Mamo

正如@FantasyFang 在他的回答中提到的,使用 lifecycle:lifecycle-extensions 的最新版本,此时是 2.2.0-alpha03。因此,您应该在 build.gradle 文件中添加以下行:

implementation 'androidx.lifecycle:lifecycle-extensions:2.2.0-alpha03' 

对于使用 Java 的用户,要解决此问题,请将这些参数直接传递给 ViewModelProvider 的 constructor

MyViewModel viewModel = new ViewModelProvider(this, myViewModelFactory).get(MyViewModel.class);

或者,如果您不使用工厂,只需使用:

MyViewModel viewModel = new ViewModelProvider(this).get(MyViewModel.class);

不传递您的工厂对象。


您应该在 build.gradle 中使用以下内容: implementation "androidx.lifecycle:lifecycle-viewmodel:$lifecycle_version" 如 developer.android.com/jetpack/androidx/releases/lifecycle 中所述,生命周期扩展中的 API 已被弃用。相反,为您需要的特定生命周期工件添加依赖项。
B
Braian Coronel

进口

弃用自:


 
  import androidx.lifecycle.ViewModelProviders;
 

至:

import androidx.lifecycle.ViewModelProvider;

使用

弃用自:


 
  ViewModelProviders.of
 (this, provider).get(VM::class.java)

至:

ViewModelProvider(this, provider).get(VM::class.java)

正是我想要的。
完美的。这应该是公认的答案。
K
Kenneth Argo

从 2.2.0 开始。生命周期扩展已被弃用。请参阅Google Documentation

这是页面的剪辑:

生命周期扩展中的 API 已被弃用。相反,为您需要的特定生命周期工件添加依赖项。

新图书馆是:

// ViewModel and lifecycle support for java
implementation "androidx.lifecycle:lifecycle-viewmodel:${versions.lifecycle}"
implementation "androidx.lifecycle:lifecycle-livedata:${versions.lifecycle}"

// ViewModel and lifecycle support for kotlin
implementation "androidx.lifecycle:lifecycle-viewmodel-ktx:${versions.lifecycle}"
implementation "androidx.lifecycle:lifecycle-livedata-ktx:${versions.lifecycle}"

JAVA的新代码:

viewModel = new ViewModelProvider(this).get(MyViewModel.class);

或者对于 Kotlin:

viewModel = ViewModelProvider(this).get(MyViewModel::class.java)

我们是保留 ViewModelProvider 还是必须重新创建它,在这种情况下,所有底层实例也将被销毁。如果我的想法是观察片段/活动的生命周期,我不是重新创建它的原因
感谢您包含 Java——我们中的一些人仍然需要维护遗留代码。
C
CommonsWare

2020-06-16 更新目前 ViewModelProviders 已弃用,不应再使用。这个问题和答案来自 2018 年底,当时情况并非如此。此问题和答案也适用于 ViewModelProviders 的旧架构组件版本,而不是 AndroidX 版本。

使用最新的依赖项 android.arch.lifecycle:extensions:1.1.1 时没有这样的类 ViewModelProviders。

就在这里。为了证明这一点:

在 Android Studio 3.2.1 中创建一个新项目(使用 Kotlin,minSdkVersion 21,“空活动”模板)

将 android.arch.lifecycle:extensions:1.1.1 添加到 app 模块的依赖项中

这会给你一个 app/build.gradle 像:

apply plugin: 'com.android.application'

apply plugin: 'kotlin-android'

apply plugin: 'kotlin-android-extensions'

android {
    compileSdkVersion 28
    defaultConfig {
        applicationId "com.commonsware.myandroidarch"
        minSdkVersion 21
        targetSdkVersion 28
        versionCode 1
        versionName "1.0"
        testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
    }
    buildTypes {
        release {
            minifyEnabled false
            proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
        }
    }
}

dependencies {
    implementation fileTree(dir: 'libs', include: ['*.jar'])
    implementation"org.jetbrains.kotlin:kotlin-stdlib-jdk7:$kotlin_version"
    implementation 'com.android.support:appcompat-v7:28.0.0'
    implementation 'com.android.support.constraint:constraint-layout:1.1.3'
    implementation 'android.arch.lifecycle:extensions:1.1.1'
    testImplementation 'junit:junit:4.12'
    androidTestImplementation 'com.android.support.test:runner:1.0.2'
    androidTestImplementation 'com.android.support.test.espresso:espresso-core:3.0.2'
}

然后,您将看到该库与该类一起显示在“外部库”中:

https://i.stack.imgur.com/h7aFI.png

您将能够引用该类:

package com.commonsware.myandroidarch

import android.arch.lifecycle.ViewModelProviders
import android.support.v7.app.AppCompatActivity
import android.os.Bundle

class MainActivity : AppCompatActivity() {

  override fun onCreate(savedInstanceState: Bundle?) {
    super.onCreate(savedInstanceState)
    setContentView(R.layout.activity_main)

    val provider = ViewModelProviders.of(this)
  }
}

转到 ViewModelProviders 的文档,我看到一条评论说:此类在 API 级别 1.1.0 中已弃用。使用 ViewModelProvider.AndroidViewModelFactory

该注释位于 ViewModelProviders.DefaultFactory 类条目下方,指的是该类,而不是 ViewModelProviders

https://i.stack.imgur.com/DDhQE.png

有什么想法可以替换上面不推荐使用的代码吗?

使用 ViewModelProviders


@TareKKhoury:这就解释了!跟踪哪些模块需要哪些依赖项肯定是一个挑战。我很高兴听到你让它工作了!
这不再是真的,ViewModelProviders.of() 不再存在。
A
Azhagthott

2020 年初,Google 在 androidx 生命周期库的 2.2.0 版本中弃用了 ViewModelProviders 类。

不再需要使用 ViewModelProviders 来创建 ViewModel 的实例,您可以将 Fragment 或 Activity 实例传递给 ViewModelProvider 构造函数。

如果您使用如下代码:

val viewModel = ViewModelProviders.of(this).get(CalculatorViewModel::class.java)

您将收到一条警告说 ViewModelProviders 已被弃用。

为避免使用已弃用的库,请进行以下更改:

在 build.gradle (Module: app) 文件中,使用 2.2.0 版本的生命周期组件: implementation 'androidx.lifecycle:lifecycle-extensions:2.2.0' implementation "androidx.activity:activity-ktx:1.1.0"如果您想使用 Fragment 中的 ViewModel,请使用 implementation "androidx.fragment:fragment-ktx:1.2.2" fragment-ktx 自动包含 activity-ktx,因此您无需在依赖项中同时指定两者。您需要在 android 部分指定 Java 8: android { compileSdkVersion 28 defaultConfig { applicationId "com.kgandroid.calculator" minSdkVersion 17 targetSdkVersion 28 versionCode 1 versionName "1.0" testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner" } buildTypes { release { minifyEnabled false proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' } } kotlinOptions { jvmTarget = "1.8" } } 在您的 Fragment 或 Activity 中,将导入更改为: import androidx.activity.viewModels创建 ViewModel 的代码变为: val viewModel: CalculatorViewModel by viewModels() 而不是 val viewModel = ViewModelProviders.of(this).get(CalculatorViewModel::class.java) 使用 viewModel 对象: val viewModel: CalculatorViewModel by viewModels( ) viewModel.newNumber.observe(this, Observer { stringResult -> newNumber.setText(stringResult) }) 其中 newNumer 是一个 LiveData 对象 在您想要共享 Activity 视图的 Fragment 中模型,您将使用 val viewModel: CalculatorViewModel by activityViewModels()

这相当于在(已弃用)ViewModelProviders.of() 函数中传递 Activity 实例。


J
Junaid

https://i.stack.imgur.com/WSw5g.png

直接使用 ViewModelProvider 构造函数,因为它们现在处理默认的 ViewModelProvider.Factory 角色。

 mainActivityViewModel = new ViewModelProvider(this).get(MainActivityViewModel.class);

x
xiaoyu

可能你可以使用:

val viewModel = ViewModelProvider(this, ViewModelProvider.NewInstanceFactory()).get(MyViewModel::class.java)

无需添加 android.arch.lifecycle:extensions:1.1.1 作为依赖项。


H
Hayk Mkrtchyan

实际上,ViewModelProviders.of() 方法在幕后做了什么?

@Deprecated
@NonNull
@MainThread
public static ViewModelProvider of(@NonNull Fragment fragment) {
    return new ViewModelProvider(fragment);
}

它将 Fragment 作为参数,创建 ViewModelProvider 对象并将片段直接传递给 ViewModelProvider 构造函数。

我们也可以用同样的方法。

例如之前:

OurViewModel mOurViewModel = ViewModelProviders.of(this).get(OurViewModel.class);

后:

OurViewModel mOurViewModel = new ViewModelProvider(this).get(OurViewModel.class);

如果您实现工厂,您还有另一种方法:ViewModelProvider 有一个构造函数,它需要 ViewModelStoreOwnerFactory
P
Paulo Linhares - Packapps

是的@Tarek,它已被弃用。现在与 AndroidX 一起使用:

val yourViewModel = ViewModelProvider.NewInstanceFactory().create(YourVideoModel::class.java)

这个答案是错误的。您应该使用 ViewModelProvider(viewModelStoreOwner, factory).get(YourVideoModel::class.java),否则您的 onCleared() 将无法正常工作,并且 ViewModel 不会在配置更改时保留。 不要使用这个答案。
I
Ilyas

直接使用 ViewModelProvider 而不是文档中提到的用户 ViewModelProviders.of() 。

ViewModelProvider(this).get(XViewModel::class.java)

https://developer.android.com/reference/androidx/lifecycle/ViewModelProviders


F
F.Mysir

如果您观看来自 Udacity 的视频并在其中创建了 GameViewModel,并且您不想编写已弃用的代码,则只需替换:

viewModel = ViewModelProviders.of(this).get(GameViewModel::class.java)

使用以下代码:

viewModel = ViewModelProvider(this).get(GameViewModel::class.java)
  

并尽快返回阅读!


I
Igor Fridman

我正在使用 android X 并且也遇到了这个问题。

首先,您应该将这些依赖项添加到您的 Gradle

implementation "androidx.lifecycle:lifecycle-extensions:$lifecycle_version"
kapt "androidx.lifecycle:lifecycle-compiler:$lifecycle_version" 

就我而言,$lifecycle_version 是 2.2.0-rc02

第二: ViewModelProvider 的导入应该是:

import androidx.lifecycle.ViewModelProvider

您可以像下面的示例一样初始化您的 viewModel:

val viewModel = ViewModelProvider(this, YourFactoryInstace).get(MainViewModel::class.java)

val viewModel = ViewModelProvider(this).get(MainViewModel::class.java)

谢谢您的回答,但是当我使用它时,viewmodel 观察和字段数据绑定不再起作用
我相信 androidx.lifecycle:lifecycle-compiler 已被 androidx.lifecycle:lifecycle-common-java8 淘汰。
L
LCZ

如果您使用 Kotlin,您可以像这样使用属性委托 viewModels()

val viewModel: YourViewModel by viewModels()

来源:https://forums.bignerdranch.com/t/solution-to-deprecated-method-viewmodelproviders-of/16833


D
Daniel Schütte

用这个:

YourViewModel yourViewModel = new ViewModelProvider.AndroidViewModelFactory(getApplication()).create(YourViewModel.class);

每次调用创建时,这不会创建新实例吗?假设像这样的 lifcyle , stop , onCreate ?
A
Amin

您可以改用它:

viewModel= ViewModelProvider(this).get(YourViewModel::class.java)

括号中的“this”是 YourViewModel 实例的所有者。


Z
ZeePee

我不是专家,但我有一行代码可以解决我在 Java 中的问题。希望这可以帮助某人。

viewModel = 
new ViewModelProvider
.AndroidViewModelFactory(getApplication())
.create(ViewModel.class);

正如我所说,我很乐意提供更多细节,但这为我解决了这个问题。


此代码错误,onCleared() 不会以这种方式调用,并且每次都会重新创建您的 ViewModel(不会为配置更改而保留)。
你能用一个特定的例子解释一下,什么可以解决我的错误?请粘贴对解释的引用或任何可以使事情更清楚的内容
viewModel = new ViewModelProvider(this).get(MyViewModel.class)。如果您的 AndroidViewModel 没有获得 Application,则说明 Lifecycle 和 Activity/Fragment 之间的版本不匹配。
好的,所以调用 getApplicationContext 是错误的?它应该只是getContext吗?
问题是创建 new ViewModelProvider.AndroidViewModelFactory( 而不是 new ViewModelProvider(
C
Caner

此答案适用于 Java,但您可以理解要点。解决方案是您需要使用 ViewModelProvider 因为ViewModelProviders deprecated:

//Do not forget import it.
import androidx.lifecycle.AndroidViewModel;

ViewModelProvider.AndroidViewModelFactory(getApplication()).create(YourViewModel.class);

示例用法是:

YourViewModel yourViewModel = new ViewModelProvider.AndroidViewModelFactory(getApplication()).create(YourViewModel.class);

另外,不要忘记更新 Gradle 依赖项Check here for last versions of them

implementation 'androidx.lifecycle:lifecycle-viewmodel:2.2.0'
annotationProcessor 'androidx.lifecycle:lifecycle-compiler:2.2.0'

!!!但是,要小心,因为一些答案推荐了这个解决方案,但它对我不起作用:

yourViewModel = new ViewModelProvider(this).get(YourViewModel.class);

因为当我使用这种方式时,我收到以下错误消息:

原因:java.lang.RuntimeException:无法创建类 com.canerkaseler.mvvmexample.ViewModel 的实例


这个答案是错误的。您需要将工厂传递给 ViewModelProvider。如果您直接调用 create,则 onCleared() 将无法正常工作。
实际上,您实际上只是说不要使用正确的答案。听起来您的 androidx.coreandroidx.lifecycle 依赖项之间不匹配。
通常我将工厂写入 ViewModelProvider。但是,您删除了它。 ContactViewModel contactViewModel = new ViewModelProvider.AndroidViewModelFactory(getApplication()).create(ContactViewModel.class);你为什么在我的回答中删除它?你传递了错误的代码。
A
Apoorv Vardhman

当您使用匕首时,您将在构造函数中传递工厂

  MobileViewModel mobileViewModel = new ViewModelProvider(this,providerFactory).get(MobileViewModel.class);

佚名

我一直在尝试并最终在 Kotlin 中使用了这个解决方案,你应该也可以在 java 中尝试它。

MainActivity.kt

 val provider : ViewModelProvider = ViewModelProvider(this)
 val VM = provider.get(ViewModelClass::class.java)

构建.gradle

implementation "androidx.lifecycle:lifecycle-viewmodel-ktx:2.3.0"
implementation "androidx.lifecycle:lifecycle-livedata-ktx:2.3.0"

截至 2021 年 2 月 2 日,生命周期库的当前版本为 2.3.0,请参阅 https://developer.android.com/jetpack/androidx/releases/lifecycle 以检查当前版本。


M
Mostafa Amer

我正在使用工厂。

ViewModelProvider.AndroidViewModelFactory
            .getInstance(getApplication())
            .create(DashboardViewModel.class);

R
Ramesh R

只需从更新

ViewModelProviders.of(this, provider).get(MainActivityViewModel::class.java)

ViewModelProvider(this, provider).get(MainActivityViewModel::class.java)


H
Heshan Sandeepa

我面临着同样的问题。如果有什么不起作用,请尝试将这段代码放入您的 oncreate 并根据您的文件进行编辑,然后繁荣您就是 gtg

val model = ViewModelProvider(viewModelStore,ViewModelProvider.AndroidViewModelFactory.getInstance(application)).get(MainActivityViewModel::class.java)

// 你只需要改变模型类


请阅读 How to answer 并更新您的问题。
D
Deepak Rajput

你可以使用这样的东西:

private  val sharedViewModel: SharedViewModel by viewModels(ownerProducer = { requireActivity() })


ownerProducer = { requireActivity() } // This is only required if you want to share same view model among different fragments.

P
Pratik Dodiya

我正在使用生命周期库 2.3.1 并在 build.gradle(模块级)中添加以下依赖项

implementation 'androidx.lifecycle:lifecycle-viewmodel-ktx:2.3.1'
implementation 'androidx.lifecycle:lifecycle-livedata-ktx:2.3.1'
implementation 'androidx.lifecycle:lifecycle-runtime-ktx:2.3.1'
implementation 'androidx.lifecycle:lifecycle-extensions-ktx:2.2.0'
implementation 'androidx.activity:activity-ktx:1.4.0'
implementation 'androidx.fragment:fragment-ktx:1.3.6'

在您的 Kotlin 代码中声明 viewModels(),如下所示:

import androidx.fragment.app.viewModels

private val cartViewModel: CartViewModel by viewModels()

T
Tashidev

更新:2022 年 7 月 20 日

私有lateinit var绑定:FragmentHomeBinding私有lateinit var homeMvvM:HomeViewModel

override fun onCreate(savedInstanceState: Bundle?) {
    super.onCreate(savedInstanceState)
    homeMvvM = ViewModelProvider(this)[HomeViewModel::class.java]

这对我有用。


S
Shrawan

尝试这个...

 LocationViewModel locationViewModel = new ViewModelProvider(this).get(LocationViewModel.class);

E
Eddy Yoel Fresno Hernández

如果你有这个实现

'implementation 'androidx.lifecycle:lifecycle-extensions:2.2.0'

尝试使用 viewModel = MainViewModel()


这不是实例化视图模型的方法。您必须为此使用 ViewModelProvider 类
H
Hamdi Hanbali

它应该以这种方式工作

 viewModel = ViewModelProviders.of(this@MainActivity).get(MainActivityViewModel::class.java)

虽然您的代码将实例化该类,但不会创建与 View Model 类正确响应生命周期事件所需的视图状态的连接。您需要以正确的方式实例化视图模型,以便它们响应并尊重其父级的生命周期,否则它们每次都会被重新创建为新鲜和空白。
如上所述,OP 和其他人指出; ViewModelProviders 类已被弃用。请参阅:developer.android.com/reference/androidx/lifecycle/…

关注公众号,不定期副业成功案例分享
关注公众号

不定期副业成功案例分享

领先一步获取最新的外包任务吗?

立即订阅