Koin DI 在发布签名的 apk 时崩溃
Koin DI crashes with release signed apk
我有一个使用 Koin DI 框架的应用程序。我已经为改造、数据库、存储库等设置了所有模块,并且在调试模式下工作正常。最近我将它上传到 Play 商店,但我发现当我尝试启动该应用程序时它崩溃了。问题是它无法为我的存储库 class 创建实例,这当然有其依赖性。
这是我在 Logcat
上遇到的错误
2020-09-09 17:23:37.690 19662-19662/? E/AndroidRuntime: FATAL EXCEPTION: main
Process: com.mypackage.myapp, PID: 19662
java.lang.RuntimeException: Unable to start activity ComponentInfo{com.mypackage.myapp/com.mypackage.ui.activities.accounts.accountlist.AccountsListActivity}: org.koin.core.error.InstanceCreationException: Could not create instance for [Single:'com.mypackage.repositories.server.ServerRepository']
at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:3632)
at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:3784)
at android.app.servertransaction.LaunchActivityItem.execute(LaunchActivityItem.java:83)
at android.app.servertransaction.TransactionExecutor.executeCallbacks(TransactionExecutor.java:135)
at android.app.servertransaction.TransactionExecutor.execute(TransactionExecutor.java:95)
at android.app.ActivityThread$H.handleMessage(ActivityThread.java:2270)
at android.os.Handler.dispatchMessage(Handler.java:107)
at android.os.Looper.loop(Looper.java:237)
at android.app.ActivityThread.main(ActivityThread.java:8125)
at java.lang.reflect.Method.invoke(Native Method)
at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:496)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1100)
Caused by: org.koin.core.error.InstanceCreationException: Could not create instance for [Single:'com.mypackage.repositories.server.ServerRepository']
at org.koin.core.instance.InstanceFactory.create(InstanceFactory.kt:59)
at org.koin.core.instance.SingleInstanceFactory.create(SingleInstanceFactory.kt:40)
at org.koin.core.instance.SingleInstanceFactory.get(SingleInstanceFactory.kt:48)
at org.koin.core.registry.InstanceRegistry.resolveInstance$koin_core(InstanceRegistry.kt:87)
at org.koin.core.scope.Scope.resolveInstance(Scope.kt:214)
at org.koin.core.scope.Scope.get(Scope.kt:181)
at com.mypackage.ui.activities.accounts.accountlist.AccountsViewModel$$special$$inlined$inject.invoke(KoinComponent.kt:67)
at kotlin.UnsafeLazyImpl.getValue(Lazy.kt:81)
at com.mypackage.ui.activities.accounts.accountlist.AccountsViewModel.getServerRepository(Unknown Source:2)
at com.mypackage.ui.activities.accounts.accountlist.AccountsViewModel.getServerAccounts(AccountsViewModel.kt:37)
at com.mypackage.ui.activities.accounts.accountlist.AccountsListActivity.onCreate(AccountsListActivity.kt:97)
at android.app.Activity.performCreate(Activity.java:7957)
at android.app.Activity.performCreate(Activity.java:7946)
at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1307)
at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:3607)
at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:3784)
at android.app.servertransaction.LaunchActivityItem.execute(LaunchActivityItem.java:83)
at android.app.servertransaction.TransactionExecutor.executeCallbacks(TransactionExecutor.java:135)
at android.app.servertransaction.TransactionExecutor.execute(TransactionExecutor.java:95)
at android.app.ActivityThread$H.handleMessage(ActivityThread.java:2270)
at android.os.Handler.dispatchMessage(Handler.java:107)
at android.os.Looper.loop(Looper.java:237)
at android.app.ActivityThread.main(ActivityThread.java:8125)
at java.lang.reflect.Method.invoke(Native Method)
at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:496)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1100)
Caused by: org.koin.core.error.InstanceCreationException: Could not create instance for [Single:'com.mypackage.managers.api.ApiManager']
at org.koin.core.instance.InstanceFactory.create(InstanceFactory.kt:59)
at org.koin.core.instance.SingleInstanceFactory.create(SingleInstanceFactory.kt:40)
at org.koin.core.instance.SingleInstanceFactory.get(SingleInstanceFactory.kt:48)
at org.koin.core.registry.InstanceRegistry.resolveInstance$koin_core(InstanceRegistry.kt:87)
at org.koin.core.scope.Scope.resolveInstance(Scope.kt:214)
at org.koin.core.scope.Scope.get(Scope.kt:181)
at com.mypackage.di.ModulesKt$repositoryModule.invoke(Modules.kt:208)
at com.mypackage.di.ModulesKt$repositoryModule.invoke(Unknown Source:4)
at org.koin.core.instance.InstanceFactory.create(InstanceFactory.kt:50)
at org.koin.core.instance.SingleInstanceFactory.create(SingleInstanceFactory.kt:40)
at org.koin.core.instance.SingleInstanceFactory.get(SingleInstanceFactory.kt:48)
at org.koin.core.registry.InstanceRegistry.resolveInstance$koin_core(InstanceRegistry.kt:87)
at org.koin.core.scope.Scope.resolveInstance(Scope.kt:214)
at org.koin.core.scope.Scope.get(Scope.kt:181)
at com.mypackage.ui.activities.accounts.accountlist.AccountsViewModel$$special$$inlined$inject.invoke(KoinComponent.kt:67)
at kotlin.UnsafeLazyImpl.getValue(Lazy.kt:81)
at com.mypackage.ui.activities.accounts.accountlist.AccountsViewModel.getServerRepository(Unknown Source:2)
at com.mypackage.ui.activities.accounts.accountlist.AccountsViewModel.getServerAccounts(AccountsViewModel.kt:37)
at com.mypackage.ui.activities.accounts.accountlist.AccountsListActivity.onCreate(AccountsListActivity.kt:97)
at android.app.Activity.performCreate(Activity.java:7957)
at android.app.Activity.performCreate(Activity.java:7946)
at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1307)
at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:3607)
at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:3784)
at android.app.servertransaction.LaunchActivityItem.execute(LaunchActivityItem.java:83)
at android.app.servertransaction.TransactionExecutor.executeCallbacks(TransactionExecutor.java:135)
at android.app.servertransaction.TransactionExecutor.execute(TransactionExecutor.java:95)
at android.app.ActivityThread$H.handleMessage(ActivityThread.java:2270)
at android.os.Handler.dispatchMessage(Handler.java:107)
at android.os.Looper.loop(Looper.java:237)
at android.app.ActivityThread.main(ActivityThread.java:8125)
at java.lang.reflect.Method.invoke(Native Method)
at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:496)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1100)
Caused by: org.koin.core.error.InstanceCreationException: Could not create instance for [Single:'retrofit2.Retrofit$Builder']
at org.koin.core.instance.InstanceFactory.create(InstanceFactory.kt:59)
at org.koin.core.instance.SingleInstanceFactory.create(SingleInstanceFactory.kt:40)
at org.koin.core.instance.SingleInstanceFactory.get(SingleInstanceFactory.kt:48)
at org.koin.core.registry.InstanceRegistry.resolveInstance$koin_core(InstanceRegistry.kt:87)
at org.koin.core.scope.Scope.resolveInstance(Scope.kt:214)
2020-09-09 17:23:37.691 19662-19662/? E/AndroidRuntime: at org.koin.core.scope.Scope.get(Scope.kt:181)
at com.mypackage.di.ModulesKt$managersModule.invoke(Modules.kt:204)
at com.mypackage.di.ModulesKt$managersModule.invoke(Unknown Source:4)
at org.koin.core.instance.InstanceFactory.create(InstanceFactory.kt:50)
... 33 more
Caused by: org.koin.core.error.InstanceCreationException: Could not create instance for [Single:'okhttp3.OkHttpClient']
at org.koin.core.instance.InstanceFactory.create(InstanceFactory.kt:59)
at org.koin.core.instance.SingleInstanceFactory.create(SingleInstanceFactory.kt:40)
at org.koin.core.instance.SingleInstanceFactory.get(SingleInstanceFactory.kt:48)
at org.koin.core.registry.InstanceRegistry.resolveInstance$koin_core(InstanceRegistry.kt:87)
at org.koin.core.scope.Scope.resolveInstance(Scope.kt:214)
at org.koin.core.scope.Scope.get(Scope.kt:181)
at com.mypackage.di.ModulesKt$networkModule.invoke(Modules.kt:204)
at com.mypackage.di.ModulesKt$networkModule.invoke(Unknown Source:4)
at org.koin.core.instance.InstanceFactory.create(InstanceFactory.kt:50)
... 41 more
Caused by: java.lang.IllegalStateException: Single instance created couldn't return value
at org.koin.core.instance.SingleInstanceFactory.get(SingleInstanceFactory.kt:50)
at org.koin.core.registry.InstanceRegistry.resolveInstance$koin_core(InstanceRegistry.kt:87)
at org.koin.core.scope.Scope.resolveInstance(Scope.kt:214)
at org.koin.core.scope.Scope.get(Scope.kt:181)
at com.mypackage.di.ModulesKt$networkModule.invoke(Modules.kt:200)
at com.mypackage.di.ModulesKt$networkModule.invoke(Unknown Source:4)
at org.koin.core.instance.InstanceFactory.create(InstanceFactory.kt:50)
... 49 more
我的build.gradle在下面
apply plugin: 'com.android.application'
apply plugin: 'kotlin-android'
apply plugin: 'kotlin-android-extensions'
apply plugin: 'kotlin-kapt'
apply plugin: "androidx.navigation.safeargs"
apply plugin: 'com.google.gms.google-services'
apply plugin: 'com.google.firebase.crashlytics'
android {
compileSdkVersion 30
defaultConfig {
applicationId "com.mypackage"
minSdkVersion 21
targetSdkVersion 30
versionCode 1
versionName "1.0"
testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"
lintOptions {
checkReleaseBuilds false
}
resConfigs "en"
}
buildTypes {
release {
minifyEnabled false
proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'
}
}
kotlinOptions {
jvmTarget = '1.8'
}
compileOptions {
sourceCompatibility JavaVersion.VERSION_1_8
targetCompatibility JavaVersion.VERSION_1_8
}
flavorDimensions "version"
productFlavors {
myapp {
dimension "version"
applicationIdSuffix ".myapp"
versionCode 2
versionName "1.2"
}
}
}
dependencies {
implementation fileTree(dir: 'libs', include: ['*.jar'])
implementation "org.jetbrains.kotlin:kotlin-stdlib-jdk7:$kotlin_version"
implementation 'androidx.appcompat:appcompat:1.2.0'
implementation 'androidx.core:core-ktx:1.3.1'
implementation 'androidx.legacy:legacy-support-v4:1.0.0'
implementation 'androidx.constraintlayout:constraintlayout:2.0.1'
implementation 'com.google.android.material:material:1.3.0-alpha02'
implementation 'androidx.recyclerview:recyclerview:1.2.0-alpha05'
implementation 'androidx.cardview:cardview:1.0.0'
implementation 'io.github.luizgrp.sectionedrecyclerviewadapter:sectionedrecyclerviewadapter:1.2.0'
implementation 'androidx.exifinterface:exifinterface:1.2.0'
// Firebase && Crashlytics
implementation 'com.google.firebase:firebase-core:17.5.0'
implementation 'com.google.firebase:firebase-messaging:20.2.4'
implementation 'com.google.firebase:firebase-crashlytics:17.2.1'
implementation 'com.google.firebase:firebase-analytics:17.5.0'
// Lifecycle (ViewModel & LiveData)
def lifecycle_version = "2.2.0"
implementation "androidx.lifecycle:lifecycle-viewmodel-ktx:$lifecycle_version"
implementation "androidx.lifecycle:lifecycle-extensions:$lifecycle_version"
implementation "androidx.lifecycle:lifecycle-livedata-ktx:$lifecycle_version"
kapt "androidx.lifecycle:lifecycle-compiler:$lifecycle_version"
// Gson to Kotlin Object
implementation 'com.google.code.gson:gson:2.8.6'
// Coroutines for Kotlin
def coroutines_version = "1.3.5"
implementation "org.jetbrains.kotlinx:kotlinx-coroutines-core:$coroutines_version"
implementation "org.jetbrains.kotlinx:kotlinx-coroutines-android:$coroutines_version"
// Room Database
def room_version = "2.2.5"
implementation "androidx.room:room-runtime:$room_version"
implementation "androidx.room:room-ktx:$room_version"
kapt "androidx.room:room-compiler:$room_version"
// Koin DI
def koin_version = "2.1.6"
implementation "org.koin:koin-core:$koin_version"
implementation "org.koin:koin-android:$koin_version"
// Navigation Components
def nav_version = "2.3.0"
implementation "androidx.navigation:navigation-fragment-ktx:$nav_version"
implementation "androidx.navigation:navigation-ui-ktx:$nav_version"
// Butterknife
def butterknifeVersion = '10.2.0'
implementation "com.jakewharton:butterknife:10.2.3"
kapt "com.jakewharton:butterknife-compiler:10.2.3"
// Preferences
implementation "androidx.preference:preference:1.1.1"
// Work Manager (Kotlin + coroutines)
implementation "androidx.work:work-runtime-ktx:2.4.0"
// Paging
implementation 'androidx.paging:paging-runtime-ktx:2.1.2'
// Circle Image view
implementation 'de.hdodenhof:circleimageview:3.1.0'
// Gallery picker
implementation 'com.kroegerama:bottomsheet-imagepicker:1.1.2'
// Sliding up Panel
implementation 'com.sothree.slidinguppanel:library:3.4.0'
// Material Search bar
implementation 'com.github.mancj:MaterialSearchBar:0.8.2'
// Material About page
implementation 'com.github.jrvansuita:MaterialAbout:0.2.3'
// Material Spinner
implementation 'com.github.chivorns.androidx:smartmaterialspinner:1.2.1'
// Pin Lock View
implementation 'com.chaos.view:pinview:1.4.3'
// Image editor
implementation 'com.github.iamutkarshtiwari:Ananas:1.2.4'
// Volley SyncRequests
implementation 'com.android.volley:volley:1.1.1'
// Retrofit Requests
implementation 'com.squareup.retrofit2:retrofit:2.9.0'
implementation 'com.squareup.retrofit2:converter-gson:2.9.0'
implementation 'com.jakewharton.retrofit:retrofit2-kotlin-coroutines-adapter:0.9.2'
implementation 'com.squareup.retrofit2:converter-simplexml:2.9.0'
// Retrofit logging interceptor
implementation 'com.squareup.okhttp3:logging-interceptor:4.8.1'
// Glide for ImageLoading
implementation 'com.github.bumptech.glide:glide:4.11.0'
kapt 'com.github.bumptech.glide:compiler:4.11.0'
// Video Compress
implementation "com.github.AbedElazizShe:LightCompressor:0.7.0"
// Qr Code scanner
implementation 'me.dm7.barcodescanner:zxing:1.9.13'
testImplementation 'junit:junit:4.13'
androidTestImplementation 'androidx.test.ext:junit:1.1.2'
androidTestImplementation 'androidx.test.espresso:espresso-core:3.3.0'
// Twilio
implementation "com.twilio:video-android:5.6.0"
implementation "com.twilio:audioswitch:0.1.0"
implementation "com.twilio:twilio-android-env:1.0.0"
// OpenCv for Grayscale video
// implementation "com.quickbirdstudios:opencv:3.4.1"
// -----DEBUG MODE--- //
// Database monitor --for Emulator we run "adb forward tcp:8080 tcp:8080"
debugImplementation 'com.amitshekhar.android:debug-db:1.0.6'
// Logback
implementation 'org.slf4j:slf4j-api:1.7.30'
implementation 'com.github.tony19:logback-android:2.0.0'
}
Koin 的模块声明如下
import androidx.room.Room
import com.google.gson.Gson
import com.google.gson.GsonBuilder
import com.jakewharton.retrofit2.adapter.kotlin.coroutines.CoroutineCallAdapterFactory
import okhttp3.Interceptor
import okhttp3.OkHttpClient
import okhttp3.logging.HttpLoggingInterceptor
import org.koin.android.ext.koin.androidContext
import org.koin.dsl.module
import retrofit2.Retrofit
import retrofit2.converter.gson.GsonConverterFactory
import java.util.concurrent.TimeUnit
val networkModule = module {
single<Gson> {
val gsonBuilder = GsonBuilder()
return@single gsonBuilder.setLenient().create()
}
single<Interceptor?> {
if (!BuildConfig.DEBUG) return@single null
val httpLoggingInterceptor = HttpLoggingInterceptor()
httpLoggingInterceptor.level = HttpLoggingInterceptor.Level.BODY
return@single httpLoggingInterceptor
}
single {
return@single HeaderInterceptor()
}
single {
return@single NetworkResponseAdapterFactory()
}
single {
return@single BaseUrlHolder(BuildConfig.BASE_HOLDER_URL)
}
single<OkHttpClient> {
val loggingInterceptor = get<Interceptor?>()
val httpClientBuilder = OkHttpClient().newBuilder().apply {
connectTimeout(60, TimeUnit.SECONDS)
readTimeout(60, TimeUnit.SECONDS)
writeTimeout(60, TimeUnit.SECONDS)
addInterceptor(get<HeaderInterceptor>())
loggingInterceptor?.let {
addInterceptor(loggingInterceptor)
}
}
return@single httpClientBuilder.build()
}
single<Retrofit.Builder> {
return@single Retrofit.Builder()
.addCallAdapterFactory(CoroutineCallAdapterFactory())
.addConverterFactory(GsonConverterFactory.create(get()))
.client(get<OkHttpClient>())
}
single<MyAppApiService> {
return@single get<Retrofit>().create(MyAppApiService::class.java)
}
}
val databaseModule = module {
single {
return@single BuildConfig.DATABASE_NAME
}
single {
return@single Room
.databaseBuilder(androidContext(), AppDatabase::class.java, get())
.build()
}
single {
get<AppDatabase>().serverDao()
}
single {
get<AppDatabase>().accountsDao()
}
single {
get<AppDatabase>().siteDao()
}
single {
get<AppDatabase>().accountGroupsDao()
}
single {
get<AppDatabase>().siteRolesDao()
}
single {
get<AppDatabase>().formsDao()
}
single {
get<AppDatabase>().messageDao()
}
single {
get<AppDatabase>().messageQueueDao()
}
}
val repositoryModule = module {
single<ServerRepository> {
return@single ServerRepositoryImpl(get(), get(), get())
}
single<AccountsRepository> {
return@single AccountsRepositoryImpl(
get(),
get(),
get(),
get(),
get(),
get(),
get(),
get(),
get(),
androidContext(),
get()
)
}
single<FormsRepository> {
return@single FormsRepositoryImpl(get())
}
single<MessagesRepository> {
return@single MessagesRepositoryImpl(get(), get(), get(), get(), get(), get(), androidContext())
}
single<FriendShipRepository> {
return@single FriendShipRepositoryImpl(get(), get(), get())
}
}
val managersModule = module {
single<SharedPrefsData<Settings>> {
return@single SharedPrefsDataImpl<Settings>(androidContext(), Settings.prefsName)
}
single<SharedPrefsManager> {
return@single SharedPrefsManagerImpl(androidContext(), get())
}
single<ApiManager> {
return@single ApiManagerImpl(get(), get())
}
single<RoomManager> {
return@single RoomManagerImpl(androidContext(), get(), get())
}
}
val lifecycleModule = module {
single {
return@single MyAppLifeCycleObserver()
}
}
既然 Koin 设置正确(因为它在调试模式下工作),为什么会发生这种情况???
问题已解决。
似乎 httpLoggingInterceptor 的提供程序 return NULL 如果它是调试模式则无法创建单个。所以我将代码更改为
single<Interceptor> {
val httpLoggingInterceptor = HttpLoggingInterceptor()
httpLoggingInterceptor.level = HttpLoggingInterceptor.Level.BODY
return@single httpLoggingInterceptor
}
.....
single {
val httpClientBuilder = OkHttpClient().newBuilder().apply {
connectTimeout(60, TimeUnit.SECONDS)
readTimeout(60, TimeUnit.SECONDS)
writeTimeout(60, TimeUnit.SECONDS)
addInterceptor(get<HeaderInterceptor>())
if (!BuildConfig.DEBUG) addInterceptor(get<Interceptor>())
}
return@single httpClientBuilder.build()
}
现在可以了。
我有一个使用 Koin DI 框架的应用程序。我已经为改造、数据库、存储库等设置了所有模块,并且在调试模式下工作正常。最近我将它上传到 Play 商店,但我发现当我尝试启动该应用程序时它崩溃了。问题是它无法为我的存储库 class 创建实例,这当然有其依赖性。 这是我在 Logcat
上遇到的错误2020-09-09 17:23:37.690 19662-19662/? E/AndroidRuntime: FATAL EXCEPTION: main
Process: com.mypackage.myapp, PID: 19662
java.lang.RuntimeException: Unable to start activity ComponentInfo{com.mypackage.myapp/com.mypackage.ui.activities.accounts.accountlist.AccountsListActivity}: org.koin.core.error.InstanceCreationException: Could not create instance for [Single:'com.mypackage.repositories.server.ServerRepository']
at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:3632)
at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:3784)
at android.app.servertransaction.LaunchActivityItem.execute(LaunchActivityItem.java:83)
at android.app.servertransaction.TransactionExecutor.executeCallbacks(TransactionExecutor.java:135)
at android.app.servertransaction.TransactionExecutor.execute(TransactionExecutor.java:95)
at android.app.ActivityThread$H.handleMessage(ActivityThread.java:2270)
at android.os.Handler.dispatchMessage(Handler.java:107)
at android.os.Looper.loop(Looper.java:237)
at android.app.ActivityThread.main(ActivityThread.java:8125)
at java.lang.reflect.Method.invoke(Native Method)
at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:496)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1100)
Caused by: org.koin.core.error.InstanceCreationException: Could not create instance for [Single:'com.mypackage.repositories.server.ServerRepository']
at org.koin.core.instance.InstanceFactory.create(InstanceFactory.kt:59)
at org.koin.core.instance.SingleInstanceFactory.create(SingleInstanceFactory.kt:40)
at org.koin.core.instance.SingleInstanceFactory.get(SingleInstanceFactory.kt:48)
at org.koin.core.registry.InstanceRegistry.resolveInstance$koin_core(InstanceRegistry.kt:87)
at org.koin.core.scope.Scope.resolveInstance(Scope.kt:214)
at org.koin.core.scope.Scope.get(Scope.kt:181)
at com.mypackage.ui.activities.accounts.accountlist.AccountsViewModel$$special$$inlined$inject.invoke(KoinComponent.kt:67)
at kotlin.UnsafeLazyImpl.getValue(Lazy.kt:81)
at com.mypackage.ui.activities.accounts.accountlist.AccountsViewModel.getServerRepository(Unknown Source:2)
at com.mypackage.ui.activities.accounts.accountlist.AccountsViewModel.getServerAccounts(AccountsViewModel.kt:37)
at com.mypackage.ui.activities.accounts.accountlist.AccountsListActivity.onCreate(AccountsListActivity.kt:97)
at android.app.Activity.performCreate(Activity.java:7957)
at android.app.Activity.performCreate(Activity.java:7946)
at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1307)
at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:3607)
at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:3784)
at android.app.servertransaction.LaunchActivityItem.execute(LaunchActivityItem.java:83)
at android.app.servertransaction.TransactionExecutor.executeCallbacks(TransactionExecutor.java:135)
at android.app.servertransaction.TransactionExecutor.execute(TransactionExecutor.java:95)
at android.app.ActivityThread$H.handleMessage(ActivityThread.java:2270)
at android.os.Handler.dispatchMessage(Handler.java:107)
at android.os.Looper.loop(Looper.java:237)
at android.app.ActivityThread.main(ActivityThread.java:8125)
at java.lang.reflect.Method.invoke(Native Method)
at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:496)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1100)
Caused by: org.koin.core.error.InstanceCreationException: Could not create instance for [Single:'com.mypackage.managers.api.ApiManager']
at org.koin.core.instance.InstanceFactory.create(InstanceFactory.kt:59)
at org.koin.core.instance.SingleInstanceFactory.create(SingleInstanceFactory.kt:40)
at org.koin.core.instance.SingleInstanceFactory.get(SingleInstanceFactory.kt:48)
at org.koin.core.registry.InstanceRegistry.resolveInstance$koin_core(InstanceRegistry.kt:87)
at org.koin.core.scope.Scope.resolveInstance(Scope.kt:214)
at org.koin.core.scope.Scope.get(Scope.kt:181)
at com.mypackage.di.ModulesKt$repositoryModule.invoke(Modules.kt:208)
at com.mypackage.di.ModulesKt$repositoryModule.invoke(Unknown Source:4)
at org.koin.core.instance.InstanceFactory.create(InstanceFactory.kt:50)
at org.koin.core.instance.SingleInstanceFactory.create(SingleInstanceFactory.kt:40)
at org.koin.core.instance.SingleInstanceFactory.get(SingleInstanceFactory.kt:48)
at org.koin.core.registry.InstanceRegistry.resolveInstance$koin_core(InstanceRegistry.kt:87)
at org.koin.core.scope.Scope.resolveInstance(Scope.kt:214)
at org.koin.core.scope.Scope.get(Scope.kt:181)
at com.mypackage.ui.activities.accounts.accountlist.AccountsViewModel$$special$$inlined$inject.invoke(KoinComponent.kt:67)
at kotlin.UnsafeLazyImpl.getValue(Lazy.kt:81)
at com.mypackage.ui.activities.accounts.accountlist.AccountsViewModel.getServerRepository(Unknown Source:2)
at com.mypackage.ui.activities.accounts.accountlist.AccountsViewModel.getServerAccounts(AccountsViewModel.kt:37)
at com.mypackage.ui.activities.accounts.accountlist.AccountsListActivity.onCreate(AccountsListActivity.kt:97)
at android.app.Activity.performCreate(Activity.java:7957)
at android.app.Activity.performCreate(Activity.java:7946)
at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1307)
at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:3607)
at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:3784)
at android.app.servertransaction.LaunchActivityItem.execute(LaunchActivityItem.java:83)
at android.app.servertransaction.TransactionExecutor.executeCallbacks(TransactionExecutor.java:135)
at android.app.servertransaction.TransactionExecutor.execute(TransactionExecutor.java:95)
at android.app.ActivityThread$H.handleMessage(ActivityThread.java:2270)
at android.os.Handler.dispatchMessage(Handler.java:107)
at android.os.Looper.loop(Looper.java:237)
at android.app.ActivityThread.main(ActivityThread.java:8125)
at java.lang.reflect.Method.invoke(Native Method)
at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:496)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1100)
Caused by: org.koin.core.error.InstanceCreationException: Could not create instance for [Single:'retrofit2.Retrofit$Builder']
at org.koin.core.instance.InstanceFactory.create(InstanceFactory.kt:59)
at org.koin.core.instance.SingleInstanceFactory.create(SingleInstanceFactory.kt:40)
at org.koin.core.instance.SingleInstanceFactory.get(SingleInstanceFactory.kt:48)
at org.koin.core.registry.InstanceRegistry.resolveInstance$koin_core(InstanceRegistry.kt:87)
at org.koin.core.scope.Scope.resolveInstance(Scope.kt:214)
2020-09-09 17:23:37.691 19662-19662/? E/AndroidRuntime: at org.koin.core.scope.Scope.get(Scope.kt:181)
at com.mypackage.di.ModulesKt$managersModule.invoke(Modules.kt:204)
at com.mypackage.di.ModulesKt$managersModule.invoke(Unknown Source:4)
at org.koin.core.instance.InstanceFactory.create(InstanceFactory.kt:50)
... 33 more
Caused by: org.koin.core.error.InstanceCreationException: Could not create instance for [Single:'okhttp3.OkHttpClient']
at org.koin.core.instance.InstanceFactory.create(InstanceFactory.kt:59)
at org.koin.core.instance.SingleInstanceFactory.create(SingleInstanceFactory.kt:40)
at org.koin.core.instance.SingleInstanceFactory.get(SingleInstanceFactory.kt:48)
at org.koin.core.registry.InstanceRegistry.resolveInstance$koin_core(InstanceRegistry.kt:87)
at org.koin.core.scope.Scope.resolveInstance(Scope.kt:214)
at org.koin.core.scope.Scope.get(Scope.kt:181)
at com.mypackage.di.ModulesKt$networkModule.invoke(Modules.kt:204)
at com.mypackage.di.ModulesKt$networkModule.invoke(Unknown Source:4)
at org.koin.core.instance.InstanceFactory.create(InstanceFactory.kt:50)
... 41 more
Caused by: java.lang.IllegalStateException: Single instance created couldn't return value
at org.koin.core.instance.SingleInstanceFactory.get(SingleInstanceFactory.kt:50)
at org.koin.core.registry.InstanceRegistry.resolveInstance$koin_core(InstanceRegistry.kt:87)
at org.koin.core.scope.Scope.resolveInstance(Scope.kt:214)
at org.koin.core.scope.Scope.get(Scope.kt:181)
at com.mypackage.di.ModulesKt$networkModule.invoke(Modules.kt:200)
at com.mypackage.di.ModulesKt$networkModule.invoke(Unknown Source:4)
at org.koin.core.instance.InstanceFactory.create(InstanceFactory.kt:50)
... 49 more
我的build.gradle在下面
apply plugin: 'com.android.application'
apply plugin: 'kotlin-android'
apply plugin: 'kotlin-android-extensions'
apply plugin: 'kotlin-kapt'
apply plugin: "androidx.navigation.safeargs"
apply plugin: 'com.google.gms.google-services'
apply plugin: 'com.google.firebase.crashlytics'
android {
compileSdkVersion 30
defaultConfig {
applicationId "com.mypackage"
minSdkVersion 21
targetSdkVersion 30
versionCode 1
versionName "1.0"
testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"
lintOptions {
checkReleaseBuilds false
}
resConfigs "en"
}
buildTypes {
release {
minifyEnabled false
proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'
}
}
kotlinOptions {
jvmTarget = '1.8'
}
compileOptions {
sourceCompatibility JavaVersion.VERSION_1_8
targetCompatibility JavaVersion.VERSION_1_8
}
flavorDimensions "version"
productFlavors {
myapp {
dimension "version"
applicationIdSuffix ".myapp"
versionCode 2
versionName "1.2"
}
}
}
dependencies {
implementation fileTree(dir: 'libs', include: ['*.jar'])
implementation "org.jetbrains.kotlin:kotlin-stdlib-jdk7:$kotlin_version"
implementation 'androidx.appcompat:appcompat:1.2.0'
implementation 'androidx.core:core-ktx:1.3.1'
implementation 'androidx.legacy:legacy-support-v4:1.0.0'
implementation 'androidx.constraintlayout:constraintlayout:2.0.1'
implementation 'com.google.android.material:material:1.3.0-alpha02'
implementation 'androidx.recyclerview:recyclerview:1.2.0-alpha05'
implementation 'androidx.cardview:cardview:1.0.0'
implementation 'io.github.luizgrp.sectionedrecyclerviewadapter:sectionedrecyclerviewadapter:1.2.0'
implementation 'androidx.exifinterface:exifinterface:1.2.0'
// Firebase && Crashlytics
implementation 'com.google.firebase:firebase-core:17.5.0'
implementation 'com.google.firebase:firebase-messaging:20.2.4'
implementation 'com.google.firebase:firebase-crashlytics:17.2.1'
implementation 'com.google.firebase:firebase-analytics:17.5.0'
// Lifecycle (ViewModel & LiveData)
def lifecycle_version = "2.2.0"
implementation "androidx.lifecycle:lifecycle-viewmodel-ktx:$lifecycle_version"
implementation "androidx.lifecycle:lifecycle-extensions:$lifecycle_version"
implementation "androidx.lifecycle:lifecycle-livedata-ktx:$lifecycle_version"
kapt "androidx.lifecycle:lifecycle-compiler:$lifecycle_version"
// Gson to Kotlin Object
implementation 'com.google.code.gson:gson:2.8.6'
// Coroutines for Kotlin
def coroutines_version = "1.3.5"
implementation "org.jetbrains.kotlinx:kotlinx-coroutines-core:$coroutines_version"
implementation "org.jetbrains.kotlinx:kotlinx-coroutines-android:$coroutines_version"
// Room Database
def room_version = "2.2.5"
implementation "androidx.room:room-runtime:$room_version"
implementation "androidx.room:room-ktx:$room_version"
kapt "androidx.room:room-compiler:$room_version"
// Koin DI
def koin_version = "2.1.6"
implementation "org.koin:koin-core:$koin_version"
implementation "org.koin:koin-android:$koin_version"
// Navigation Components
def nav_version = "2.3.0"
implementation "androidx.navigation:navigation-fragment-ktx:$nav_version"
implementation "androidx.navigation:navigation-ui-ktx:$nav_version"
// Butterknife
def butterknifeVersion = '10.2.0'
implementation "com.jakewharton:butterknife:10.2.3"
kapt "com.jakewharton:butterknife-compiler:10.2.3"
// Preferences
implementation "androidx.preference:preference:1.1.1"
// Work Manager (Kotlin + coroutines)
implementation "androidx.work:work-runtime-ktx:2.4.0"
// Paging
implementation 'androidx.paging:paging-runtime-ktx:2.1.2'
// Circle Image view
implementation 'de.hdodenhof:circleimageview:3.1.0'
// Gallery picker
implementation 'com.kroegerama:bottomsheet-imagepicker:1.1.2'
// Sliding up Panel
implementation 'com.sothree.slidinguppanel:library:3.4.0'
// Material Search bar
implementation 'com.github.mancj:MaterialSearchBar:0.8.2'
// Material About page
implementation 'com.github.jrvansuita:MaterialAbout:0.2.3'
// Material Spinner
implementation 'com.github.chivorns.androidx:smartmaterialspinner:1.2.1'
// Pin Lock View
implementation 'com.chaos.view:pinview:1.4.3'
// Image editor
implementation 'com.github.iamutkarshtiwari:Ananas:1.2.4'
// Volley SyncRequests
implementation 'com.android.volley:volley:1.1.1'
// Retrofit Requests
implementation 'com.squareup.retrofit2:retrofit:2.9.0'
implementation 'com.squareup.retrofit2:converter-gson:2.9.0'
implementation 'com.jakewharton.retrofit:retrofit2-kotlin-coroutines-adapter:0.9.2'
implementation 'com.squareup.retrofit2:converter-simplexml:2.9.0'
// Retrofit logging interceptor
implementation 'com.squareup.okhttp3:logging-interceptor:4.8.1'
// Glide for ImageLoading
implementation 'com.github.bumptech.glide:glide:4.11.0'
kapt 'com.github.bumptech.glide:compiler:4.11.0'
// Video Compress
implementation "com.github.AbedElazizShe:LightCompressor:0.7.0"
// Qr Code scanner
implementation 'me.dm7.barcodescanner:zxing:1.9.13'
testImplementation 'junit:junit:4.13'
androidTestImplementation 'androidx.test.ext:junit:1.1.2'
androidTestImplementation 'androidx.test.espresso:espresso-core:3.3.0'
// Twilio
implementation "com.twilio:video-android:5.6.0"
implementation "com.twilio:audioswitch:0.1.0"
implementation "com.twilio:twilio-android-env:1.0.0"
// OpenCv for Grayscale video
// implementation "com.quickbirdstudios:opencv:3.4.1"
// -----DEBUG MODE--- //
// Database monitor --for Emulator we run "adb forward tcp:8080 tcp:8080"
debugImplementation 'com.amitshekhar.android:debug-db:1.0.6'
// Logback
implementation 'org.slf4j:slf4j-api:1.7.30'
implementation 'com.github.tony19:logback-android:2.0.0'
}
Koin 的模块声明如下
import androidx.room.Room
import com.google.gson.Gson
import com.google.gson.GsonBuilder
import com.jakewharton.retrofit2.adapter.kotlin.coroutines.CoroutineCallAdapterFactory
import okhttp3.Interceptor
import okhttp3.OkHttpClient
import okhttp3.logging.HttpLoggingInterceptor
import org.koin.android.ext.koin.androidContext
import org.koin.dsl.module
import retrofit2.Retrofit
import retrofit2.converter.gson.GsonConverterFactory
import java.util.concurrent.TimeUnit
val networkModule = module {
single<Gson> {
val gsonBuilder = GsonBuilder()
return@single gsonBuilder.setLenient().create()
}
single<Interceptor?> {
if (!BuildConfig.DEBUG) return@single null
val httpLoggingInterceptor = HttpLoggingInterceptor()
httpLoggingInterceptor.level = HttpLoggingInterceptor.Level.BODY
return@single httpLoggingInterceptor
}
single {
return@single HeaderInterceptor()
}
single {
return@single NetworkResponseAdapterFactory()
}
single {
return@single BaseUrlHolder(BuildConfig.BASE_HOLDER_URL)
}
single<OkHttpClient> {
val loggingInterceptor = get<Interceptor?>()
val httpClientBuilder = OkHttpClient().newBuilder().apply {
connectTimeout(60, TimeUnit.SECONDS)
readTimeout(60, TimeUnit.SECONDS)
writeTimeout(60, TimeUnit.SECONDS)
addInterceptor(get<HeaderInterceptor>())
loggingInterceptor?.let {
addInterceptor(loggingInterceptor)
}
}
return@single httpClientBuilder.build()
}
single<Retrofit.Builder> {
return@single Retrofit.Builder()
.addCallAdapterFactory(CoroutineCallAdapterFactory())
.addConverterFactory(GsonConverterFactory.create(get()))
.client(get<OkHttpClient>())
}
single<MyAppApiService> {
return@single get<Retrofit>().create(MyAppApiService::class.java)
}
}
val databaseModule = module {
single {
return@single BuildConfig.DATABASE_NAME
}
single {
return@single Room
.databaseBuilder(androidContext(), AppDatabase::class.java, get())
.build()
}
single {
get<AppDatabase>().serverDao()
}
single {
get<AppDatabase>().accountsDao()
}
single {
get<AppDatabase>().siteDao()
}
single {
get<AppDatabase>().accountGroupsDao()
}
single {
get<AppDatabase>().siteRolesDao()
}
single {
get<AppDatabase>().formsDao()
}
single {
get<AppDatabase>().messageDao()
}
single {
get<AppDatabase>().messageQueueDao()
}
}
val repositoryModule = module {
single<ServerRepository> {
return@single ServerRepositoryImpl(get(), get(), get())
}
single<AccountsRepository> {
return@single AccountsRepositoryImpl(
get(),
get(),
get(),
get(),
get(),
get(),
get(),
get(),
get(),
androidContext(),
get()
)
}
single<FormsRepository> {
return@single FormsRepositoryImpl(get())
}
single<MessagesRepository> {
return@single MessagesRepositoryImpl(get(), get(), get(), get(), get(), get(), androidContext())
}
single<FriendShipRepository> {
return@single FriendShipRepositoryImpl(get(), get(), get())
}
}
val managersModule = module {
single<SharedPrefsData<Settings>> {
return@single SharedPrefsDataImpl<Settings>(androidContext(), Settings.prefsName)
}
single<SharedPrefsManager> {
return@single SharedPrefsManagerImpl(androidContext(), get())
}
single<ApiManager> {
return@single ApiManagerImpl(get(), get())
}
single<RoomManager> {
return@single RoomManagerImpl(androidContext(), get(), get())
}
}
val lifecycleModule = module {
single {
return@single MyAppLifeCycleObserver()
}
}
既然 Koin 设置正确(因为它在调试模式下工作),为什么会发生这种情况???
问题已解决。 似乎 httpLoggingInterceptor 的提供程序 return NULL 如果它是调试模式则无法创建单个。所以我将代码更改为
single<Interceptor> {
val httpLoggingInterceptor = HttpLoggingInterceptor()
httpLoggingInterceptor.level = HttpLoggingInterceptor.Level.BODY
return@single httpLoggingInterceptor
}
.....
single {
val httpClientBuilder = OkHttpClient().newBuilder().apply {
connectTimeout(60, TimeUnit.SECONDS)
readTimeout(60, TimeUnit.SECONDS)
writeTimeout(60, TimeUnit.SECONDS)
addInterceptor(get<HeaderInterceptor>())
if (!BuildConfig.DEBUG) addInterceptor(get<Interceptor>())
}
return@single httpClientBuilder.build()
}
现在可以了。