我们最近将主要的Application类转换为Kotlin。 从那时起,我们遇到了崩溃,特别是在夜间(当我们的应用程序可能被系统杀死时),当我们的JobService是startet时。
我们以静态方式访问应用程序上下文以获取某些依赖项,这些依赖项在将类转换为Kotlin之前运行良好。从那以后,静态吸气剂是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控制台中收到了用户的这些崩溃消息:
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没有初始化。
那我的问题呢?为什么转换到Kotlin时Application.onCreate()
的执行时间会发生变化?为什么在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
)
)
}
}
Jobservice
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>
SearchState
public class SearchState extends State {
//Koin Crash in SearchState.<init>
private PlacemarkRepository placemarkRepository = get(PlacemarkRepository.class);
...
}
作业服务必须有自己独立的上下文,但我认为您应该尝试升级到WorkManager,因为作业服务不适用于Android O和更高版本的设备。工作经理是一个新事物,但请记住,它允许执行工作的最小间隔是15分钟。