我们最近将我们的主要应用程序类转换为 Kotlin。
从那以后,我们遇到了崩溃,尤其是在夜间(当我们的应用程序可能被系统杀死时),当我们的 JobService 启动时。
在将类转换为 Kotlin 之前,我们正在以静态方式访问应用程序上下文以获取一些运行良好的依赖项。从那时起,静态 getter 是在应用程序函数 lateinit var
中初始化的。onCreate
发布后 Google Play 报告了这些崩溃:
Caused by: kotlin.UninitializedPropertyAccessException:
at x.y.z.application.App.access$getAppContext$cp
[...]
at x.y.z.jobs.JobSchedulerService.onCreate (JobSchedulerService.java:27)
这就引出了一个问题,我们Application.onCreate()
还没有被执行吗?
我们稍微重构了 JobService 以减少静态上下文访问的数量,直到需要进行重大重构为止。之后,我们在 Google Play Console 中收到了来自用户的这些崩溃:
Caused by: kotlin.UninitializedPropertyAccessException:
at org.koin.standalone.StandAloneContext.getKoinContext (StandAloneContext.java:45)
at org.koin.java.standalone.KoinJavaComponent.get (KoinJavaComponent.java:66)
at org.koin.java.standalone.KoinJavaComponent.get$default (KoinJavaComponent.java:64)
at org.koin.java.standalone.KoinJavaComponent.get (KoinJavaComponent.java)
at x.y.z.SearchState.<init> (SearchState.java:21)
[...]
at x.y.z.jobs.JobSchedulerService.onStartJob (JobSchedulerService.java:54)
这些崩溃告诉我们同样的事情:Application.onCreate()
还没有执行,因为 Koin 没有初始化。
所以我的问题?为什么Application.onCreate()
转换为 Kotlin 时执行时间会发生变化,或者为什么在 JobService 启动之前不再创建我们的应用程序?
我的意思是,当然,我们可以重构整个应用程序依赖项以使用 JobService 本身提供的上下文,但是如果应用程序是在之后创建的并且我们仍然想使用 Koin 怎么办?我们的应用程序可能会再次崩溃,并带有AlreadyStartetException
. 如果我们的应用程序还没有“存在”,那么服务会有什么上下文?
来源(简化):
应用
abstract class App : MultiDexApplication() {
companion object {
@JvmStatic
lateinit var appContext: Context
@JvmStatic
val isDevelopment: Boolean = BuildConfig.DEBUG
// @JvmStatic
// val isDevelopment: Boolean by lazy {
// appContext.resources.getBoolean(R.bool.isDevelopment)
// }
}
override fun onCreate() {
super.onCreate()
appContext = applicationContext
startKoin(
applicationContext,
listOf(
coreModule,
sharedPrefsModule
)
)
}
}
工作服务
public class JobSchedulerService extends JobService implements OnFinishListener {
@Override
public boolean onStartJob(JobParameters params) {
if (App.isDevelopment()) { //First crash cause `isDevelopment` relied on App.appContext
...
}
this.mJobParameters = params;
this.mStateMachine = StateContext.getInstance(getApplication());
mStateMachine.setOnFinishListener(this);
mStateMachine.execute("" + params.getJobId()); //Second crash is in the first executed state auf this state Machine
return true;
}
}
清单注册
<service
android:name="x.y.z.jobs.JobSchedulerService"
android:enabled="true"
android:exported="true"
android:permission="android.permission.BIND_JOB_SERVICE">
</service>
搜索状态
public class SearchState extends State {
//Koin Crash in SearchState.<init>
private PlacemarkRepository placemarkRepository = get(PlacemarkRepository.class);
...
}