WorkManager 内置支持长时间运行的 Worker。在这种情况下,WorkManager 可以向操作系统提供一个信号,表明在执行此工作时应尽可能保持进程存活。这些 Worker 可以运行超过 10 分钟。此新功能的示例用例包括批量上传或下载(无法分块)、在本地处理 ML 模型或对应用程序 _用户_ 重要的任务。
在后台,WorkManager 会代表您管理和运行前台服务以执行 WorkRequest
,同时还会显示可配置的通知。
ListenableWorker
现在支持 setForegroundAsync()
API,而 CoroutineWorker
支持挂起的 setForeground()
API。这些 API 允许开发人员指定此 WorkRequest
是 _重要_ 的(从用户角度来看)或 _长时间运行_ 的。
从 2.3.0-alpha03
开始,WorkManager 还允许您创建一个 PendingIntent
,该 PendingIntent
可用于取消 Worker,而无需使用 createCancelPendingIntent()
API 注册新的 Android 组件。这种方法在与 setForegroundAsync()
或 setForeground()
API 结合使用时特别有用,这些 API 可用于向通知操作添加取消 Worker
的功能。
创建和管理长时间运行的 Worker
您将使用略微不同的方法,具体取决于您是在 Kotlin 还是 Java 中编码。
Kotlin
Kotlin 开发人员应使用 CoroutineWorker
。您可以使用该方法的挂起版本 setForeground()
,而不是使用 setForegroundAsync()
。
class DownloadWorker(context: Context, parameters: WorkerParameters) :
CoroutineWorker(context, parameters) {
private val notificationManager =
context.getSystemService(Context.NOTIFICATION_SERVICE) as
NotificationManager
override suspend fun doWork(): Result {
val inputUrl = inputData.getString(KEY_INPUT_URL)
?: return Result.failure()
val outputFile = inputData.getString(KEY_OUTPUT_FILE_NAME)
?: return Result.failure()
// Mark the Worker as important
val progress = "Starting Download"
setForeground(createForegroundInfo(progress))
download(inputUrl, outputFile)
return Result.success()
}
private fun download(inputUrl: String, outputFile: String) {
// Downloads a file and updates bytes read
// Calls setForeground() periodically when it needs to update
// the ongoing Notification
}
// Creates an instance of ForegroundInfo which can be used to update the
// ongoing notification.
private fun createForegroundInfo(progress: String): ForegroundInfo {
val id = applicationContext.getString(R.string.notification_channel_id)
val title = applicationContext.getString(R.string.notification_title)
val cancel = applicationContext.getString(R.string.cancel_download)
// This PendingIntent can be used to cancel the worker
val intent = WorkManager.getInstance(applicationContext)
.createCancelPendingIntent(getId())
// Create a Notification channel if necessary
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
createChannel()
}
val notification = NotificationCompat.Builder(applicationContext, id)
.setContentTitle(title)
.setTicker(title)
.setContentText(progress)
.setSmallIcon(R.drawable.ic_work_notification)
.setOngoing(true)
// Add the cancel action to the notification which can
// be used to cancel the worker
.addAction(android.R.drawable.ic_delete, cancel, intent)
.build()
return ForegroundInfo(notificationId, notification)
}
@RequiresApi(Build.VERSION_CODES.O)
private fun createChannel() {
// Create a Notification channel
}
companion object {
const val KEY_INPUT_URL = "KEY_INPUT_URL"
const val KEY_OUTPUT_FILE_NAME = "KEY_OUTPUT_FILE_NAME"
}
}
Java
使用 ListenableWorker
或 Worker
的开发人员可以调用 setForegroundAsync()
API,该 API 返回 ListenableFuture<Void>
。您还可以调用 setForegroundAsync()
来更新正在进行的 Notification
。
以下是一个长时间运行的 Worker 的简单示例,该 Worker 下载文件。此 Worker 会跟踪进度以更新正在进行的 Notification
,该通知显示下载进度。
public class DownloadWorker extends Worker {
private static final String KEY_INPUT_URL = "KEY_INPUT_URL";
private static final String KEY_OUTPUT_FILE_NAME = "KEY_OUTPUT_FILE_NAME";
private NotificationManager notificationManager;
public DownloadWorker(
@NonNull Context context,
@NonNull WorkerParameters parameters) {
super(context, parameters);
notificationManager = (NotificationManager)
context.getSystemService(NOTIFICATION_SERVICE);
}
@NonNull
@Override
public Result doWork() {
Data inputData = getInputData();
String inputUrl = inputData.getString(KEY_INPUT_URL);
String outputFile = inputData.getString(KEY_OUTPUT_FILE_NAME);
// Mark the Worker as important
String progress = "Starting Download";
setForegroundAsync(createForegroundInfo(progress));
download(inputUrl, outputFile);
return Result.success();
}
private void download(String inputUrl, String outputFile) {
// Downloads a file and updates bytes read
// Calls setForegroundAsync(createForegroundInfo(myProgress))
// periodically when it needs to update the ongoing Notification.
}
@NonNull
private ForegroundInfo createForegroundInfo(@NonNull String progress) {
// Build a notification using bytesRead and contentLength
Context context = getApplicationContext();
String id = context.getString(R.string.notification_channel_id);
String title = context.getString(R.string.notification_title);
String cancel = context.getString(R.string.cancel_download);
// This PendingIntent can be used to cancel the worker
PendingIntent intent = WorkManager.getInstance(context)
.createCancelPendingIntent(getId());
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
createChannel();
}
Notification notification = new NotificationCompat.Builder(context, id)
.setContentTitle(title)
.setTicker(title)
.setSmallIcon(R.drawable.ic_work_notification)
.setOngoing(true)
// Add the cancel action to the notification which can
// be used to cancel the worker
.addAction(android.R.drawable.ic_delete, cancel, intent)
.build();
return new ForegroundInfo(notificationId, notification);
}
@RequiresApi(Build.VERSION_CODES.O)
private void createChannel() {
// Create a Notification channel
}
}
向长时间运行的 Worker 添加前台服务类型
如果您的应用程序定位的是 Android 14(API 等级 34)或更高版本,则必须为所有长时间运行的 Worker 指定 前台服务类型。如果您的应用程序定位的是 Android 10(API 等级 29)或更高版本,并且包含需要访问位置的长时间运行的 Worker,请说明该 Worker 使用 前台服务类型 location
。
如果您的应用程序定位的是 Android 11(API 等级 30)或更高版本,并且包含需要访问相机或麦克风的长时间运行的 Worker,请分别声明 camera
或 microphone
前台服务类型。
要添加这些前台服务类型,请完成以下各部分中描述的步骤。
在应用程序清单中声明前台服务类型
在应用程序清单中声明工作线程的前台服务类型。在以下示例中,工作线程需要访问位置和麦克风。
<service android:name="androidx.work.impl.foreground.SystemForegroundService" android:foregroundServiceType="location|microphone" tools:node="merge" />
在运行时指定前台服务类型
当您调用 setForeground()
或 setForegroundAsync()
时,请确保指定 前台服务类型.
Kotlin
private fun createForegroundInfo(progress: String): ForegroundInfo { // ... return ForegroundInfo(NOTIFICATION_ID, notification, FOREGROUND_SERVICE_TYPE_LOCATION or FOREGROUND_SERVICE_TYPE_MICROPHONE) }
Java
@NonNull private ForegroundInfo createForegroundInfo(@NonNull String progress) { // Build a notification... Notification notification = ...; return new ForegroundInfo(NOTIFICATION_ID, notification, FOREGROUND_SERVICE_TYPE_LOCATION | FOREGROUND_SERVICE_TYPE_MICROPHONE); }