我正在尝试在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需要构造函数中的Application上下文,而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"
答案 0 :(得分:27)
如果您使用的是hilt
,则可能忘记了用@AndroidEntryPoint
注释您的活动
答案 1 :(得分:8)
很明显,如果我改变了
recipeViewModel = new ViewModelProvider(this).get(RecipeViewModel.class);
对此
recipeViewModel = new ViewModelProvider(this, ViewModelProvider.AndroidViewModelFactory.getInstance(this.getApplication())).get(RecipeViewModel.class);
有效。我不知道为什么有人可以解释这种解决方案?
答案 2 :(得分:1)
我看过Android源代码。如果您的活动是扩展类Activity
,而不是AppCompatActivity
,您将不会遇到此问题。类Activity
为接口HasDefaultViewModelProviderFactory
提供了一个实现,该接口为扩展AndroidViewModel的类提供了正确的工厂。不幸的是,AppCompatActivity
仍未实现该接口,因此没有正确的工厂可用。因此,您可以像以前那样提供工厂,也可以自己实现接口HasDefaultViewModelProviderFactory
。
答案 3 :(得分:0)
您可以尝试使用已过时的旧版本吗
recipeViewModel = ViewModelProviders.of(this).get(RecipeViewModel.class);
这绝对可以在我们的应用程序中使用。如果您是冒险类,还可以使用调试器并进入ViewModelProvider的get()方法,以检查它如何尝试获取viewmodel的实例。