Не удается создать экземпляр модели просмотра после использования Hilt в Android

Вот мой AllFilesListViewModel класс.

class AllFilesListViewModel @ViewModelInject constructor(
    private val pdfItemRepository: PdfItemRepository):ViewModel() {

}

Вот это PdfItemRepository класс.

@Singleton
class PdfItemRepository @Inject constructor(private val pdfItemDao: PdfItemDao){

}

За pdfItemDao. Я создал модуль с именемDatabaseModule. Ниже приведен код -

@Module
@InstallIn(ApplicationComponent::class)
object DatabaseModule {

    @Provides
    fun provideDatabase(@ApplicationContext context: Context):AppDatabase{
        return AppDatabase.getDataBase(context)
    }

    @Provides
    fun providePdfItemDao(database:AppDatabase):PdfItemDao{
        return database.pdfItemDao()
    }
}

Вот класс фрагмента AllFilesFragment.kt где я использую viewModel.

@AndroidEntryPoint
class AllFilesFragment:Fragment(){

    private lateinit var binding:AllFilesFragmentBinding
    private val viewModel by viewModels<AllFilesListViewModel>()


    override fun onCreateView(
        inflater: LayoutInflater,
        container: ViewGroup?,
        savedInstanceState: Bundle?
    ): View? {
        binding = AllFilesFragmentBinding.inflate(inflater,container,false)
        context?: return binding.root
        initThings()
        subscribeUi()
        return binding.root
    }
}

Вот это logcat файл.

06-19 19:22:20.203 23753-23753/com.emptysheet.pdfreader_autoscroll E/AndroidRuntime: FATAL EXCEPTION: main
    Process: com.emptysheet.pdfreader_autoscroll, PID: 23753
    java.lang.RuntimeException: Cannot create an instance of class com.emptysheet.pdfreader_autoscroll.homeScreen.viewModel.AllFilesListViewModel
        at androidx.lifecycle.ViewModelProvider$NewInstanceFactory.create(ViewModelProvider.java:221)
        at androidx.lifecycle.ViewModelProvider$AndroidViewModelFactory.create(ViewModelProvider.java:278)
        at androidx.lifecycle.SavedStateViewModelFactory.create(SavedStateViewModelFactory.java:106)
        at androidx.hilt.lifecycle.HiltViewModelFactory.create(HiltViewModelFactory.java:69)
        at androidx.lifecycle.AbstractSavedStateViewModelFactory.create(AbstractSavedStateViewModelFactory.java:69)
        at androidx.lifecycle.ViewModelProvider.get(ViewModelProvider.java:185)
        at androidx.lifecycle.ViewModelProvider.get(ViewModelProvider.java:150)
        at androidx.lifecycle.ViewModelLazy.getValue(ViewModelProvider.kt:54)
        at androidx.lifecycle.ViewModelLazy.getValue(ViewModelProvider.kt:41)
        at com.emptysheet.pdfreader_autoscroll.homeScreen.AllFilesFragment.getViewModel(AllFilesFragment.kt)
        at com.emptysheet.pdfreader_autoscroll.homeScreen.AllFilesFragment.subscribeUi(AllFilesFragment.kt:72)
        at com.emptysheet.pdfreader_autoscroll.homeScreen.AllFilesFragment.onCreateView(AllFilesFragment.kt:64)
        at androidx.fragment.app.Fragment.performCreateView(Fragment.java:2698)
        at androidx.fragment.app.FragmentStateManager.createView(FragmentStateManager.java:320)
        at androidx.fragment.app.FragmentManager.moveToState(FragmentManager.java:1187)
        at androidx.fragment.app.FragmentManager.moveToState(FragmentManager.java:1356)
        at androidx.fragment.app.FragmentManager.moveFragmentToExpectedState(FragmentManager.java:1434)
        at androidx.fragment.app.BackStackRecord.executeOps(BackStackRecord.java:442)
        at androidx.fragment.app.FragmentManager.executeOps(FragmentManager.java:2169)
        at androidx.fragment.app.FragmentManager.executeOpsTogether(FragmentManager.java:1992)
        at androidx.fragment.app.FragmentManager.removeRedundantOperationsAndExecute(FragmentManager.java:1947)
        at androidx.fragment.app.FragmentManager.execSingleAction(FragmentManager.java:1818)
        at androidx.fragment.app.BackStackRecord.commitNow(BackStackRecord.java:297)
        at androidx.viewpager2.adapter.FragmentStateAdapter.placeFragmentInViewHolder(FragmentStateAdapter.java:341)
        at androidx.viewpager2.adapter.FragmentStateAdapter.onViewAttachedToWindow(FragmentStateAdapter.java:276)
        at androidx.viewpager2.adapter.FragmentStateAdapter.onViewAttachedToWindow(FragmentStateAdapter.java:67)
        at androidx.recyclerview.widget.RecyclerView.dispatchChildAttached(RecyclerView.java:7556)
        at androidx.recyclerview.widget.RecyclerView$5.addView(RecyclerView.java:860)
        at androidx.recyclerview.widget.ChildHelper.addView(ChildHelper.java:107)
        at androidx.recyclerview.widget.RecyclerView$LayoutManager.addViewInt(RecyclerView.java:8601)
        at androidx.recyclerview.widget.RecyclerView$LayoutManager.addView(RecyclerView.java:8559)
        at androidx.recyclerview.widget.RecyclerView$LayoutManager.addView(RecyclerView.java:8547)
        at androidx.recyclerview.widget.LinearLayoutManager.layoutChunk(LinearLayoutManager.java:1641)
        at androidx.recyclerview.widget.LinearLayoutManager.fill(LinearLayoutManager.java:1587)
        at androidx.recyclerview.widget.LinearLayoutManager.onLayoutChildren(LinearLayoutManager.java:665)
        at androidx.recyclerview.widget.RecyclerView.dispatchLayoutStep2(RecyclerView.java:4134)
        at androidx.recyclerview.widget.RecyclerView.dispatchLayout(RecyclerView.java:3851)
        at androidx.recyclerview.widget.RecyclerView.onLayout(RecyclerView.java:4404)
        at android.view.View.layout(View.java:15689)
        at android.view.ViewGroup.layout(ViewGroup.java:5048)
        at androidx.viewpager2.widget.ViewPager2.onLayout(ViewPager2.java:527)
        at android.view.View.layout(View.java:15689)
        at android.view.ViewGroup.layout(ViewGroup.java:5048)
        at com.google.android.material.appbar.HeaderScrollingViewBehavior.layoutChild(HeaderScrollingViewBehavior.java:148)
        at com.google.android.material.appbar.V

12 ответов

Решение

Это было решено после того, как я использовал kapt 'androidx.hilt:hilt-compiler:1.0.0-alpha01'в приложении build.gradle. Я уже добавилkapt "com.google.dagger:hilt-android-compiler:2.28-alpha". Я все еще не понимал разницы между двумя BTW. Если кто знает. Пожалуйста, объясните мне это.

Такое случается со мной при использовании Hilt, и для меня причиной проблемы было то, что я забыл добавить @AndroidEntryPoint поверх класса фрагмента

Это вызвано несоответствием версий жизненного цикла AndroidX, AndroidX Core, AndroidX Activity и AndroidX Fragment.

Эфес работает только если getDefaultViewModelProviderFactory можно переопределить.

Это верно только в том случае, если этот метод действительно существует, чего нет, если ваши зависимости устарели. А именно вашandroidx.fragment ниже 1.2.0, а ваш androidx.activity ниже 1.1.0.

Используйте это, и это сработает:

implementation "androidx.appcompat:appcompat:1.1.0"
implementation "androidx.fragment:fragment-ktx:1.2.5"
implementation "androidx.core:core-ktx:1.3.0"
implementation "androidx.activity:activity:1.1.0"

implementation "androidx.lifecycle:lifecycle-viewmodel-ktx:2.2.0"
implementation "androidx.lifecycle:lifecycle-livedata-ktx:2.2.0"
implementation "androidx.lifecycle:lifecycle-runtime-ktx:2.2.0"
implementation "androidx.lifecycle:lifecycle-viewmodel-savedstate:2.2.0"

Но в настоящее время это то, что заставляет меня работать:

implementation 'com.google.dagger:dagger:2.28'
kapt 'com.google.dagger:dagger-compiler:2.28'

// hilt
implementation 'com.google.dagger:hilt-android:2.28-alpha'
kapt 'com.google.dagger:hilt-android-compiler:2.28-alpha'
kaptAndroidTest 'com.google.dagger:hilt-android-compiler:2.28-alpha'
kaptTest 'com.google.dagger:hilt-android-compiler:2.28-alpha'
implementation 'androidx.hilt:hilt-lifecycle-viewmodel:1.0.0-alpha01'
kapt 'androidx.hilt:hilt-compiler:1.0.0-alpha01'

Вместе с

 apply plugin: 'dagger.hilt.android.plugin'

А также

    classpath 'com.google.dagger:hilt-android-gradle-plugin:2.28-alpha'
    

@ViewModelInject устарел в новой версии рукояти.

Справка

Использовать HiltViewModel

      @HiltViewModel
class AllFilesListViewModel @Inject constructor(
    val pdfItemRepository: PdfItemRepository)
) : ViewModel() {

}

Когда я использую Jetpack Compose, Hilt и Compose Navigation, мой подход состоит в том, чтобы получить все зависимости в документации и убедиться, что все их версии актуальны. Ключ в том, что когда вы создаете ViewModel, вы не должны использовать = viewModel() потому что у вас есть Compose Navigation, = hiltViewModel() следует использовать вместо этого.

Jetpack Compose + ViewModel в NavGraph

Этот ответ предназначен для людей, которые используют Jetpack Compose и навигацию (NavGraph). Согласно Руководству в документации, мы должны использовать hiltViewModelвместо viewModelПример:

      dependencies {
    implementation("androidx.hilt:hilt-navigation-compose:1.0.0")
}
      // import androidx.hilt.navigation.compose.hiltViewModel

@Composable
fun MyApp() {
    NavHost(navController, startDestination = startRoute) {
        composable("example") { backStackEntry ->
            // Creates a ViewModel from the current BackStackEntry
            // Available in the androidx.hilt:hilt-navigation-compose artifact
            val viewModel = hiltViewModel<MyViewModel>()
            MyScreen(viewModel)
        }
        /* ... */
    }
}

Больше свежей информации в и навигацииисточнике

Добавлять@AndroidEntryPointначало имени класса внутри вашего фрагмента или действия.

Так:

      @AndroidEntryPoint
class YourClassNameFragment: Fragment() {
    ....
}

Я сталкивался с этой проблемой раньше, и я исправил ее, пройдя SavedStateHandle к основному конструктору модели представления.

class AuthViewModel @ViewModelInject constructor(@Assisted private val savedState: SavedStateHandle) : ViewModel()

Для тех, кто проверил все вышеперечисленные решения и все еще не работает, последняя проверка - удалить Build папка и проект, это заставит компилятор воссоздать кинжал dependency graph под капотом.

В моем случае у меня annotated моя деятельность с @AndroidEntryPointвсе еще сталкивается с той же проблемой. Я удалил свой build папка и rebuild проект, и он работает, как ожидалось.

В alpha03 используйте новую модель @HiltViewModel и обычный @Inject, как показано ниже.

      @HiltViewModel
class MyViewModel @Inject constructor(
    private val repository: Repository,
    private val savedStateHandle: SavedStateHandle
) : ViewModel(), LifecycleObserver {

    // Some code
}

Я тоже столкнулся с этой проблемой сегодня, я попробовал все возможные исправления, но удалить ошибку не удалось. Я просто публикую здесь свое решение на случай, если у кого-то в будущем возникнет такая же проблема.

В моем случае у меня есть многомодульный проект с «модулем пользовательского интерфейса», «модулем ViewModel» и «модулем вариантов использования». Ошибка с моей стороны заключалась в том, что я не импортировал все модули в модуле gradle приложения, я импортировал только модуль пользовательского интерфейса. Я нашел эту заметку на сайте разработчиков Android относительно реализации рукояти:

Примечание. Поскольку для генерации кода Hilt требуется доступ ко всем модулям Gradle, которые используют Hilt, модуль Gradle, который компилирует ваш класс Application, также должен иметь все ваши модули Hilt и классы, внедренные конструктором, в своих транзитивных зависимостях.

Когда я импортировал все модули, необходимые для создания графика DI, этот сбой исчез.

для меня разрешение при перемещении = viewModel()от функции Compose к Activity

Другие вопросы по тегам