android 使用WorkManager在特定时间安排工作

xoefb8l8  于 2022-12-21  发布在  Android
关注(0)|答案(8)|浏览(234)

WorkManager是一个库,用于将保证在满足其约束后执行的工作排队。
因此,在浏览了Constraints类之后,我没有找到任何函数来为工作添加时间约束。例如,我想在早上8:00am开始执行一项工作(该工作可以是OneTimeWorkRequestPeriodicWorkRequest两种类型中的任何一种)。我如何使用WorkManager添加约束来调度这项工作?

fcg9iug3

fcg9iug31#

遗憾的是,您目前无法在特定时间安排工作。如果您有时间紧迫的实施,则应使用AlarmManager,通过setAndAllowWhileIdle()或setExactAndAllowWhileIdle(),将可在低电耗模式下触发的闹钟设置为。
您可以使用WorkManager计划工作,初始延迟一次或定期执行,如下所示:

创建辅助类:

public class MyWorker extends Worker {
    @Override
    public Worker.WorkerResult doWork() {

        // Do the work here

        // Indicate success or failure with your return value:
        return WorkerResult.SUCCESS;

        // (Returning RETRY tells WorkManager to try this task again
        // later; FAILURE says not to try again.)
    }
}

则按如下方式调度OneTimeWorkRequest

OneTimeWorkRequest mywork=
        new OneTimeWorkRequest.Builder(MyWorker.class)
        .setInitialDelay(<duration>, <TimeUnit>)// Use this when you want to add initial delay or schedule initial work to `OneTimeWorkRequest` e.g. setInitialDelay(2, TimeUnit.HOURS)
        .build();
WorkManager.getInstance().enqueue(mywork);

您可以按如下方式设置附加约束:

// Create a Constraints that defines when the task should run
Constraints myConstraints = new Constraints.Builder()
    .setRequiresDeviceIdle(true)
    .setRequiresCharging(true)
    // Many other constraints are available, see the
    // Constraints.Builder reference
     .build();

然后创建使用这些约束的OneTimeWorkRequest

OneTimeWorkRequest mywork=
                new OneTimeWorkRequest.Builder(MyWorker.class)
     .setConstraints(myConstraints)
     .build();
WorkManager.getInstance().enqueue(mywork);

可按如下方式创建周期性工作请求:

PeriodicWorkRequest periodicWork = new PeriodicWorkRequest.Builder(MyWorker.class, 12, TimeUnit.HOURS)
                                   .build();
  WorkManager.getInstance().enqueue(periodicWork);

这将创建一个PeriodicWorkRequest,以便每12小时定期运行一次。

np8igboo

np8igboo2#

PeriodicWorkRequests现在支持2.1.0-alpha 02版本的初始延迟。您可以使用PeriodicWorkRequest.Builder上的setInitialDelay方法来设置初始延迟。
每天上午8:00的调度示例。这里我使用joda time library进行时间操作。

final int SELF_REMINDER_HOUR = 8;

    if (DateTime.now().getHourOfDay() < SELF_REMINDER_HOUR) {
        delay = new Duration(DateTime.now() , DateTime.now().withTimeAtStartOfDay().plusHours(SELF_REMINDER_HOUR)).getStandardMinutes();
    } else {
        delay = new Duration(DateTime.now() , DateTime.now().withTimeAtStartOfDay().plusDays(1).plusHours(SELF_REMINDER_HOUR)).getStandardMinutes();
    }

    PeriodicWorkRequest workRequest = new PeriodicWorkRequest.Builder(
        WorkerReminderPeriodic.class,
        24,
        TimeUnit.HOURS,
        PeriodicWorkRequest.MIN_PERIODIC_FLEX_MILLIS,
        TimeUnit.MILLISECONDS)
        .setInitialDelay(delay, TimeUnit.MINUTES)
        .addTag("send_reminder_periodic")
        .build();

    WorkManager.getInstance()
        .enqueueUniquePeriodicWork("send_reminder_periodic", ExistingPeriodicWorkPolicy.REPLACE, workRequest);
b09cbbtk

b09cbbtk3#

到目前为止,使用PeriodicWorkRequest还不可能获得精确的时间。
一个丑陋的变通方法是使用一个OneTimeWorkRequest,当它触发时,用新的计算周期设置另一个OneTimeWorkRequest,依此类推。

qnyhuwrf

qnyhuwrf4#

我可能有点晚了,但无论如何,我这样做是为了在给定的时间安排一个工作请求(有一个可选的短延迟)。你只需要从TimePicker获得时间:

public static void scheduleWork(int hour, int minute) {
    Calendar calendar = Calendar.getInstance();
    long nowMillis = calendar.getTimeInMillis();

    calendar.set(Calendar.HOUR_OF_DAY,hour);
    calendar.set(Calendar.MINUTE,minute);
    calendar.set(Calendar.SECOND,0);
    calendar.set(Calendar.MILLISECOND,0);

    if (calendar.before(Calendar.getInstance())) {
        calendar.add(Calendar.DATE, 1);
    }

    long diff = calendar.getTimeInMillis() - nowMillis;

    WorkManager mWorkManager = WorkManager.getInstance();
    Constraints constraints = new Constraints.Builder()
            .setRequiredNetworkType(NetworkType.CONNECTED)
            .build();
    mWorkManager.cancelAllWorkByTag(WORK_TAG);
    OneTimeWorkRequest mRequest = new OneTimeWorkRequest.Builder(NotificationWorker.class)
            .setConstraints(constraints)
            .setInitialDelay(diff,TimeUnit.MILLISECONDS)
            .addTag(WORK_TAG)
            .build();
    mWorkManager.enqueue(mRequest);

}
slhcrj9b

slhcrj9b5#

所有答案现在都已过时,升级到WorkManager 2.1.0-alpha 02(或更高版本)setInitialDelay()方法,该方法过去仅适用于OneTimeWorkRequest,但现在也支持PeriodicWorkRequest。

implementation "androidx.work:work-runtime:2.1.0-alpha02"

PeriodicWorkRequests现在支持初始延迟。可以使用PeriodicWorkRequest.Builder上的setInitialDelay方法设置初始延迟
快速示例:

new PeriodicWorkRequest.Builder(MyWorker.class, MY_REPEATS, TimeUnit.HOURS)
        .setInitialDelay(THE_DELAY,TimeUnit.SECONDS);
eivgtgni

eivgtgni6#

您可以使用Evernote中的AndroidJob

class NotificationJob : DailyJob() {

override fun onRunDailyJob(params: Params): DailyJobResult {
       //your job       
       return DailyJobResult.SUCCESS

}

companion object {
    val TAG = "NotificationJob"
    fun scheduleJob() {
        //scheduled between 9-10 am

        DailyJob.schedule(JobRequest.Builder(TAG), 
            TimeUnit.HOURS.toMillis(9),TimeUnit.HOURS.toMillis(10 ))
     }
   }
 }

以及通知创建器

class NotificationJobCreator : JobCreator {

        override fun create(tag: String): Job? {
           return when (tag) {
              NotificationJob.TAG ->
                NotificationJob()
              else ->
                null
           }
       }
  }

然后在应用程序类中启动

JobManager.create(this).addJobCreator(NotificationJobCreator())

Gradle依赖关系为

dependencies {
        implementation 'com.evernote:android-job:1.2.6'

        // or this with workmnager 
        implementation 'com.evernote:android-job:1.3.0-alpha08'
    }
41zrol4v

41zrol4v7#

如果您想为PeriodicWorkRequest设置初始延迟,我在这里给出了解决方案:
在Android中为周期性工作管理器设置初始延迟

camsedfj

camsedfj8#

我试过OneTimeWorkRequest,但是它很脆弱(有时候才能工作),所以我们不应该依赖它。AlarmManager是一个更好的选择。

相关问题