web-dev-qa-db-fra.com

UninitializedPropertyAccessException: la propriété lateinit n'a pas été initialisée

Je sais que cette question a reçu une réponse [Ici] ( la propriété Lateinit n’a pas été initialisée ). Mais c'était dû à un problème de bibliothèque de couteaux à beurre. Mais mon cas est différent. complètement différent. Dans mon cas, lorsque j'utilise des propriétés de dague injectées dans mon activité de base, une erreur est provoquée par: kotlin.UninitializedPropertyAccessException: la propriété lateinit n'a pas été initialisée

Mais même propriété lorsque j'utilise dans ma sous-activité (activité de connexion) cela fonctionne très bien.

par exemple. pref.setLanguage ("abc") -> cela fonctionne correctement dans l'activité de connexion mais génère une erreur dans l'activité de base

Voici mon code:

BaseActivity

abstract class BaseActivity : AppCompatActivity() {
@Inject
lateinit var pref: AppSharedPreferences
@Inject
lateinit var utils: Utils
lateinit var mCurrentLanguage: String
protected lateinit var progressBarUtils: ProgressBarUtils

override fun onCreate(savedInstanceState: Bundle?) {
    super.onCreate(savedInstanceState)
    progressBarUtils = ProgressBarUtils(this)
    mCurrentLanguage = "EN"
    pref.setSelectedLanguage(mCurrentLanguage)   
}
}

LoginActivity

class LoginActivity : BaseActivity() {
private val TAG = this.javaClass.name
lateinit var loginViewModel: LoginViewModel
@Inject
lateinit var viewModelFactory: LoginViewModelFactory


override fun onCreate(savedInstanceState: Bundle?) {
    super.onCreate(savedInstanceState)
    setContentView(R.layout.activity_login)
    AndroidInjection.inject(this)
    loginViewModel = ViewModelProviders.of(this, viewModelFactory).get(
            LoginViewModel::class.Java)
    loadData("abc", "xyz")
    observerOnLoginResult()
    observerOnLoginError()
}


private fun observerOnLoginError() {
    loginViewModel.loginError().observe(this, Observer<String> {
        progressBarUtils.cancelProgressDialog()
        if (it != null) {
            Toast.makeText(this, resources.getString(R.string.error_message) + it,
                    Toast.LENGTH_SHORT).show()
        }
    })
}

private fun observerOnLoginResult() {
    loginViewModel.loginResult().observe(this,
            Observer<LoginModel> {
                progressBarUtils.cancelProgressDialog()
                if (it != null) {
                    setData(it)
                }
            })
}

private fun setData(loginData: LoginModel) {
    pref.setCurrentUserName(loginData.data.userName)
}

private fun loadData(email: String, password: String) {
    progressBarUtils.showProgressDialog()
    val builder = StringBuilder()
    var auth = Base64.encodeToString(("$email:$password").toByteArray(), Base64.NO_WRAP)
    builder.append("Basic ")
    builder.append(auth)
    loginViewModel.getLoginData(builder.toString())
}

override fun onDestroy() {
    loginViewModel.disposeElements()
    super.onDestroy()
}

AppModule

@Module class AppModule(val app: Application) {

@Provides
@Singleton
fun provideApplication(): Application = app


@Provides
@Singleton
fun provideSharedPreferences(): AppSharedPreferences = AppSharedPreferences(app)


@Provides
@Singleton
fun provideUtils(): Utils = Utils(app)
}

Module de génération

@Module abstract class BuildersModule {

@PerActivity
@ContributesAndroidInjector(modules = arrayOf(LoginModule::class))
abstract fun contributeLoginActivity(): LoginActivity

@PerActivity
abstract fun contributeBaseActivityActivity(): BaseActivity
}

AppComponent

@Singleton @Component(
    modules = arrayOf(AndroidInjectionModule::class, AppModule::class, BuildersModule::class))

interface AppComponent { 
  fun inject(app: LotusApp)            
}
12
Tarun

Vous ne pouvez pas utiliser un objet avant de l'initialiser.

override fun onCreate(savedInstanceState: Bundle?) {
  super.onCreate(savedInstanceState) // prefs called in parents onCreate!
  setContentView(R.layout.activity_login)
  AndroidInjection.inject(this) // injection happens here

Mettez vos appels en ordre. La solution la plus simple consiste à déplacer AndroidInjection.inject(this) avant l'appel à super.onCreate(...).

18
David Medenjak