android - 使用自定义参数实现 AndroidX Worker

标签 android kotlin androidx android-workmanager

我正在从 Android Priority JobQueue library 迁移到 AndroidX WorkManager在我的项目中,但它的构建方式是 Worker应该是这样的:

TaskSchedulerWorker.kt

class TaskSchedulerWorker(
    private val workManager: WorkManager = WorkManager.getInstance()
) : TaskScheduler {

    override fun execute(useCaseWrapper: UseCaseWrapper) {
        workManager.beginUniqueWork(
            useCaseWrapper.useCaseName,
            ExistingWorkPolicy.APPEND,
            OneTimeWorkRequest.from(UseCaseWrapperWorker::class.java)
        ).enqueue()
    }

    override fun onCancel(useCaseWrapper: UseCaseWrapper) {
        workManager.cancelUniqueWork(useCaseWrapper.useCaseName)
    }
}

UseCaseWrapperWorker.kt

class UseCaseWrapperWorker(
    context: Context, 
    params: WorkerParams,
    private val useCaseWrapper: UseCaseWrapper
) : Worker(context, params) {

    override fun doWork(): Result {
        return try {
            useCaseWrapper.execute()
            Result.success()
        } catch (ex: Exception) {
            ex.printStackTrace()
            Result.failure()
        }
    }
}

重点是,如果我尝试使用这些类运行我的应用程序,我会收到下一个错误:

2019-05-09 09:34:45.784 27260-27443/com.mobileapp.debug E/WM-WorkerFactory: Could not instantiate com.mobileapp.domain.jobqueue.UseCaseWrapperWorker
    java.lang.NoSuchMethodException: com.mobileapp.domain.jobqueue.UseCaseWrapperWorker.<init> [class android.content.Context, class androidx.work.WorkerParameters]
        at java.lang.Class.getConstructor0(Class.java:2328)
        at java.lang.Class.getDeclaredConstructor(Class.java:2167)
        at androidx.work.WorkerFactory.createWorkerWithDefaultFallback(WorkerFactory.java:92)
        at androidx.work.impl.WorkerWrapper.runWorker(WorkerWrapper.java:234)
        at androidx.work.impl.WorkerWrapper.run(WorkerWrapper.java:128)
        at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1167)
        at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:641)
        at java.lang.Thread.run(Thread.java:764)
2019-05-09 09:34:45.784 27260-27443/com.mobileapp.debug E/WM-WorkerWrapper: Could not create Worker com.mobileapp.domain.jobqueue.UseCaseWrapperWorker

因为 Worker构造函数不能有两个以上的默认参数( ContextWorkerParameters )。

我无法弄清楚如何实现 UseCaseWrapperWorker具有接收 Context 的构造函数的类,WorkerParametersUseCaseWrapper ,并从 TaskSchedulerWorker 调用它与 OneTimeWorkRequest.from(UseCaseWrapperWorker::class.java)说明。


编辑:

我尝试实现 CustomWorkerFactory和一个 CustomWorker向他们传递 UseCaseWrapper 。问题是我必须初始化 WorkManagerApplication#onCreate方法( Android Developer Documentation ),并且我无权访问 UseCaseWrapper此时需要。

TaskSchedulerWorker.kt

(WorkManager init 应该位于 Application#onCreate 中)

class TaskSchedulerWorker(private val appContext: Context) : TaskScheduler {

    override fun execute(useCaseWrapper: UseCaseWrapper) {
        val workerFactory = CustomWorkerFactory(useCaseWrapper)
        val configuration = Configuration.Builder()
            .setWorkerFactory(workerFactory)
            .build()
        WorkManager.initialize(context, configuration)

        WorkManager.getInstance().beginWork(
            "uniqueWorkName",
            ExistingWorkPolicy.APPEND,
            OneTimeWorkRequest.from(CustomWorker::class.java)
        )
    }
}

CustomWorkerFactory.kt

class CustomWorkerFactory(private val useCaseWrapper: UseCaseWrapper) : WorkerFactory() {

    override fun createWorker(
        appContext: Context,
        workerClassName: String,
        workerParameters: WorkerParameters
    ): ListenableWorker? {
        return CustomWorker(appContext, workerParameters, useCaseWrapper)
    }
}

CustomWorker.kt

class CustomWorker(
    context: Context, 
    params: WorkerParameters
) : Worker(context, params) {

    private var useCaseWrapper: UseCaseWrapper? = null

    constructor(
        context: Context,
        params: WorkerParameters,
        useCaseWrapper: UseCaseWrapper
    ) : this(context, params) {
        this.useCaseWrapper = useCaseWrapper
    }

    override fun doWork(): Result {
        return useCaseWrapper?.let { useCaseWrapper ->
            try {
                useCaseWrapper.execute()
                Result.success()
            } catch (ex: Exception) {
                Result.failure()
            }
        } ?: Result.failure()
    }
}

RuntimeException抛出的是:

java.lang.RuntimeException: Unable to start activity ComponentInfo{com.mobileapp.debug/com.mobileapp.home.view.activity.HomeActivity}: java.lang.IllegalStateException: WorkManager is already initialized.  
Did you try to initialize it manually without disabling WorkManagerInitializer? See WorkManager#initialize(Context, Configuration) or the class levelJavadoc for more information.
    at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:3086)
    at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:3229)
    at android.app.servertransaction.LaunchActivityItem.execute(LaunchActivityItem.java:78)
    at android.app.servertransaction.TransactionExecutor.executeCallbacks(TransactionExecutor.java:108)
    at android.app.servertransaction.TransactionExecutor.execute(TransactionExecutor.java:68)
    at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1926)
    at android.os.Handler.dispatchMessage(Handler.java:106)
    at android.os.Looper.loop(Looper.java:214)
    at android.app.ActivityThread.main(ActivityThread.java:6981)
    at java.lang.reflect.Method.invoke(Native Method)
    at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:493)
    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1445)
Caused by: java.lang.IllegalStateException: WorkManager is already initialized.  Did you try to initialize it manually without disabling WorkManagerInitializer? 
See WorkManager#initialize(Context, Configuration) or the class levelJavadoc for more information.
    at androidx.work.impl.WorkManagerImpl.initialize(WorkManagerImpl.java:137)
    at androidx.work.WorkManager.initialize(WorkManager.java:169)
    at com.mobileapp.domain.worker.TaskSchedulerWorker.execute(TaskSchedulerWorker.kt:20)
    at com.mobileapp.domain.usecase.UseCaseHandler.execute(UseCaseHandler.kt:41)
    at com.mobileapp.domain.usecase.UseCaseCall.execute(UseCaseCall.kt:50)
    at com.mobileapp.home.view.presenter.HomePresenter.getConfigurationJSON(HomePresenter.kt:107)
    at com.mobileapp.home.view.activity.HomeActivity.initComponents(HomeActivity.kt:107)
    at com.mobileapp.home.view.activity.HomeActivity.onCreate(HomeActivity.kt:67)
    at android.app.Activity.performCreate(Activity.java:7326)
    at android.app.Activity.performCreate(Activity.java:7317)
    at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1271)
    at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:3066)
    at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:3229) 
    at android.app.servertransaction.LaunchActivityItem.execute(LaunchActivityItem.java:78) 
    at android.app.servertransaction.TransactionExecutor.executeCallbacks(TransactionExecutor.java:108) 
    at android.app.servertransaction.TransactionExecutor.execute(TransactionExecutor.java:68) 
    at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1926) 
    at android.os.Handler.dispatchMessage(Handler.java:106) 
    at android.os.Looper.loop(Looper.java:214) 
    at android.app.ActivityThread.main(ActivityThread.java:6981) 
    at java.lang.reflect.Method.invoke(Native Method) 
    at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:493) 
    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1445) 

有什么建议吗?

最佳答案

您应该使用自定义 WorkerFactory 来初始化您的 Workers。请参阅https://developer.android.com/reference/androidx/work/WorkerFactory

关于android - 使用自定义参数实现 AndroidX Worker,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/56054521/

相关文章:

Android AVD 无法识别设备但 ADB 可以

kotlin - 在协程处理程序和 try-catch block 中处理异常

Kotlin:如何从二进制转换为十进制

android - context.getSystemService(UserManager.class) 在 Android 9 华为手机上返回 null

android - 如何使用新的 Jetpack 导航在 Actionbar 中获取后退按钮

android - 将 lint 添加到 CI 并在 lint 为 Android Studio 项目产生错误时标记构建失败

android - 如何在 Android 中下载 pdf 文件?

android - Android 中的 Chrome 如何知道应用程序已安装

java - 找不到 androidx.camera :camera-view

android x 设计依赖