检查WorkRequest是否先前已被WorkManager包围

时间:2018-07-18 11:42:00

标签: android android-jetpack android-workmanager

我正在使用PeriodicWorkRequest每15分钟执行一次任务。 我想检查一下是否定期安排了这个定期工作请求。如果没有,请安排它。

     if (!PreviouslyScheduled) {
        PeriodicWorkRequest dataupdate = new PeriodicWorkRequest.Builder( DataUpdateWorker.class , 15 , TimeUnit.MINUTES).build();
        WorkManager.getInstance().enqueue(dataupdate);
      }

以前,当我使用JobScheduler执行任务时,我曾经使用

public static boolean isJobServiceScheduled(Context context, int JOB_ID ) {
    JobScheduler scheduler = (JobScheduler) context.getSystemService( Context.JOB_SCHEDULER_SERVICE ) ;

    boolean hasBeenScheduled = false ;

    for ( JobInfo jobInfo : scheduler.getAllPendingJobs() ) {
        if ( jobInfo.getId() == JOB_ID ) {
            hasBeenScheduled = true ;
            break ;
        }
    }

    return hasBeenScheduled ;
}

需要帮助为工作请求构建类似的模块,以帮助查找计划的/活动的工作请求。

4 个答案:

答案 0 :(得分:8)

为您的PeriodicWorkRequest任务设置一些标签:

    PeriodicWorkRequest work =
            new PeriodicWorkRequest.Builder(DataUpdateWorker.class, 15, TimeUnit.MINUTES)
                    .addTag(TAG)
                    .build();

然后在enqueue()工作之前检查带有TAG的任务:

    WorkManager wm = WorkManager.getInstance();
    ListenableFuture<List<WorkStatus>> future = wm.getStatusesByTag(TAG);
    List<WorkStatus> list = future.get();
    // start only if no such tasks present
    if((list == null) || (list.size() == 0)){
        // shedule the task
        wm.enqueue(work);
    } else {
        // this periodic task has been previously scheduled
    }

但是,如果您真的不需要知道它是否已预先安排,则可以使用:

    static final String TASK_ID = "data_update"; // some unique string id for the task
    PeriodicWorkRequest work =
            new PeriodicWorkRequest.Builder(DataUpdateWorker.class,
                    15, TimeUnit.MINUTES)
                    .build();

    WorkManager.getInstance().enqueueUniquePeriodicWork(TASK_ID,
                ExistingPeriodicWorkPolicy.KEEP, work);

ExistingPeriodicWorkPolicy.KEEP意味着该任务仅计划一次,然后即使在设备重新启动后也可以定期工作。如果您需要重新安排任务的时间(例如,如果您需要更改任务的某些参数),则需要在此处使用ExistingPeriodicWorkPolicy.REPLACE

答案 1 :(得分:1)

您需要为每个WorkRequest添加一个唯一标签。选中Tagged work

您可以通过将标签字符串分配给任何WorkRequest对象来对任务进行逻辑分组。为此,您需要致电WorkRequest.Builder.addTag()

检查以下Android文档示例:

OneTimeWorkRequest cacheCleanupTask =
    new OneTimeWorkRequest.Builder(MyCacheCleanupWorker.class)
.setConstraints(myConstraints)
.addTag("cleanup")
.build();

PeriodicWorkRequest相同

然后,您将使用WorkManager.getStatusesByTag()获得带有该标签的所有任务的所有WorkStatus的列表。

这会给您LiveDataWorkStatus列表,用于标记了标签的作品。

然后您可以使用WorkStatus检查状态,如下所示:

       WorkStatus workStatus = listOfWorkStatuses.get(0);

        boolean finished = workStatus.getState().isFinished();
        if (!finished) {
            // Work InProgress
        } else {
            // Work Finished
        }

您可以在下面的google示例中查看更多详细信息。他们在这里添加了如何向WorkRequest添加标签以及如何通过标签获取工作状态的方法:

https://github.com/googlecodelabs/android-workmanager

编辑 检查以下代码,并评论如何通过标签获取WorkStatus。如果WorkStatus结果为空,请安排我们的工作。

 // Check work status by TAG
    WorkManager.getInstance().getStatusesByTag("[TAG_STRING]").observe(this, listOfWorkStatuses -> {

        // Note that we will get single WorkStatus if any tag (here [TAG_STRING]) related Work exists

        // If there are no matching work statuses
        // then we make sure that periodic work request has been previously not scheduled
        if (listOfWorkStatuses == null || listOfWorkStatuses.isEmpty()) {
           // we can schedule our WorkRequest here
            PeriodicWorkRequest dataupdate = new PeriodicWorkRequest.Builder( DataUpdateWorker.class , 15 , TimeUnit.MINUTES)
                    .addTag("[TAG_STRING]")
                    .build();
            WorkManager.getInstance().enqueue(dataupdate);
            return;
        }

        WorkStatus workStatus = listOfWorkStatuses.get(0);
        boolean finished = workStatus.getState().isFinished();
        if (!finished) {
            // Work InProgress
        } else {
            // Work Finished
        }
    });

我尚未测试代码。请提供您的反馈意见。

希望这对您有所帮助。

答案 2 :(得分:0)

我也在寻找相同的条件,找不到条件,因此为了解决这个问题,我找到了一种机制,首先取消所有预定的工作并重新计划工作。这样我们可以确保仅维护您的一项工作。另外,请确保必须维护这样的工作代码逻辑。

用于取消作品。对于more

UUID compressionWorkId = compressionWork.getId();
WorkManager.getInstance().cancelWorkById(compressionWorkId);

答案 3 :(得分:0)

与ExistingPeriodicWorkPolicy保持相同的taskID。KEEP不会每次都创建新任务。

.enqueueUniquePeriodicWork(TASK_ID, ExistingPeriodicWorkPolicy.KEEP,工作);