JobScheduler JobService在没有Application的情况下启动

问题描述 投票:14回答:1

我们最近将主要的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。如果我们的应用程序尚未“存在”,那么该服务将具有哪些上下文?

Sources (simplified):

应用

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);

    ...
}
android kotlin android-service android-jobscheduler koin
1个回答
2
投票

作业服务必须有自己独立的上下文,但我认为您应该尝试升级到WorkManager,因为作业服务不适用于Android O和更高版本的设备。工作经理是一个新事物,但请记住,它允许执行工作的最小间隔是15分钟。

© www.soinside.com 2019 - 2024. All rights reserved.