我正试图在MainActivity中创建我的AndroidViewModel实例。当我这样做时,我会得到以下错误没有零参数构造函数
这是我的RecipeViewModel
package com.example.kookrecepten;
import android.app.Application;
import androidx.lifecycle.AndroidViewModel;
import androidx.lifecycle.LiveData;
import java.util.List;
public class RecipeViewModel extends AndroidViewModel {
private RecipeRepository recipeRepository;
private LiveData<List<Recipe>> allRecipes;
public RecipeViewModel(Application application) {
super(application);
recipeRepository = new RecipeRepository(application);
allRecipes = recipeRepository.getAllRecipes();
}
public void insert(Recipe recipe) {
recipeRepository.insert(recipe);
}
public void update(Recipe recipe) {
recipeRepository.update(recipe);
}
public void delete(Recipe recipe) {
recipeRepository.delete(recipe);
}
public void deleteAll() {
recipeRepository.deleteAllRecipes();
}
public LiveData<List<Recipe>> getAllRecipes() {
return allRecipes;
}
}
如果我错了,现在纠正我的错误,但AndroidViewModel需要构造函数中的应用程序上下文,而ViewModel不需要。所以我不知道为什么android要求一个零参数构造函数。
这是我请求一个实例的主要活动。
package com.example.kookrecepten;
import android.os.Bundle;
import android.view.View;
import android.widget.Toast;
import androidx.appcompat.app.AppCompatActivity;
import androidx.lifecycle.Observer;
import androidx.lifecycle.ViewModelProvider;
import com.example.kookrecepten.databinding.ActivityMainBinding;
import java.util.List;
public class MainActivity extends AppCompatActivity {
private ActivityMainBinding binding;
private RecipeViewModel recipeViewModel;
protected void onCreate (Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
binding = ActivityMainBinding.inflate(getLayoutInflater());
View view = binding.getRoot();
setContentView(view);
//Get an instance of the RecipeViewModel.
recipeViewModel = new ViewModelProvider(this).get(RecipeViewModel.class);
recipeViewModel.getAllRecipes().observe(this, new Observer<List<Recipe>>() {
@Override
public void onChanged(List<Recipe> recipes) {
//Update recycle view
Toast.makeText(MainActivity.this, "triggered", Toast.LENGTH_SHORT).show();
}
});
}
}
这是我的实现。
def lifecycle_version = "2.2.0"
def room_version = "2.2.5"
//LifeCycle Components
// ViewModel
implementation "androidx.lifecycle:lifecycle-viewmodel:$lifecycle_version"
// LiveData
implementation "androidx.lifecycle:lifecycle-livedata:$lifecycle_version"
// Annotation processor
implementation "androidx.lifecycle:lifecycle-common-java8:$lifecycle_version"
//Room Components
implementation "androidx.room:room-runtime:$room_version"
annotationProcessor "androidx.room:room-compiler:$room_version"
如果您使用hilt
,您可能忘记用@AndroidEntryPoint
注释您的活动
如果我更改
recipeViewModel = new ViewModelProvider(this).get(RecipeViewModel.class);
到这个
recipeViewModel = new ViewModelProvider(this, ViewModelProvider.AndroidViewModelFactory.getInstance(this.getApplication())).get(RecipeViewModel.class);
它有效。我不知道为什么这个解决方案有效,有人能解释吗?
我查看了Android源代码。如果你的活动是扩展类Activity
,而不是AppCompatActivity
,你就不会看到这个问题。类Activity
提供了接口HasDefaultViewModelProviderFactory
的实现,该接口为扩展AndroidViewModel的类提供了正确的工厂。不幸的是,AppCompatActivity
仍然没有实现该接口,并且没有正确的工厂可用。因此,您可以像以前那样提供工厂,也可以自己实现接口HasDefaultViewModelProviderFactory
。
你能试用旧的、不推荐使用的吗
recipeViewModel = ViewModelProviders.of(this).get(RecipeViewModel.class);
这在我们的应用程序中确实适用。如果你是冒险型,你也可以使用调试器并进入ViewModelProvider的get((方法来检查它是如何尝试获取视图模型的实例的。