В определенных ситуациях вам может потребоваться предоставить собственную стратегию потоковой обработки. Например, вам может потребоваться обработать асинхронную операцию на основе обратного вызова. WorkManager поддерживает этот вариант использования с помощью ListenableWorker
. ListenableWorker
— это самый простой рабочий API; Worker
, CoroutineWorker
и RxWorker
являются производными от этого класса. ListenableWorker
сигнализирует только о том, когда работа должна начинаться и останавливаться, и оставляет управление потоками полностью на ваше усмотрение. Сигнал начала работы вызывается в основном потоке, поэтому очень важно вручную перейти к фоновому потоку по вашему выбору.
Абстрактный метод ListenableWorker.startWork()
возвращает ListenableFuture
Result
. ListenableFuture
— это облегченный интерфейс: это Future
, предоставляющий функциональные возможности для подключения прослушивателей и распространения исключений. Ожидается, что в методе startWork
вы вернете ListenableFuture
, который вы установите вместе с Result
операции после ее завершения. Экземпляры ListenableFuture
можно создавать одним из двух способов:
- Если вы используете Guava, используйте
ListeningExecutorService
. - В противном случае включите
councurrent-futures
в свой файл градиента и используйтеCallbackToFutureAdapter
.
Если вы хотите выполнить какую-то работу на основе асинхронного обратного вызова, вы должны сделать что-то вроде этого:
Котлин
class CallbackWorker( context: Context, params: WorkerParameters ) : ListenableWorker(context, params) { override fun startWork(): ListenableFuture<Result> { return CallbackToFutureAdapter.getFuture { completer -> val callback = object : Callback { var successes = 0 override fun onFailure(call: Call, e: IOException) { completer.setException(e) } override fun onResponse(call: Call, response: Response) { successes++ if (successes == 100) { completer.set(Result.success()) } } } repeat(100) { downloadAsynchronously("https://example.com", callback) } callback } } }
Ява
public class CallbackWorker extends ListenableWorker { public CallbackWorker(Context context, WorkerParameters params) { super(context, params); } @NonNull @Override public ListenableFuture<Result> startWork() { return CallbackToFutureAdapter.getFuture(completer -> { Callback callback = new Callback() { int successes = 0; @Override public void onFailure(Call call, IOException e) { completer.setException(e); } @Override public void onResponse(Call call, Response response) { successes++; if (successes == 100) { completer.set(Result.success()); } } }; for (int i = 0; i < 100; i++) { downloadAsynchronously("https://www.example.com", callback); } return callback; }); } }
Что произойдет, если вашу работу остановят ? ListenableFuture
объекта ListenableWorker
всегда отменяется, когда ожидается остановка работы. Используя CallbackToFutureAdapter
, вам просто нужно добавить прослушиватель отмены, как показано ниже:
Котлин
class CallbackWorker( context: Context, params: WorkerParameters ) : ListenableWorker(context, params) { override fun startWork(): ListenableFuture<Result> { return CallbackToFutureAdapter.getFuture { completer -> val callback = object : Callback { var successes = 0 override fun onFailure(call: Call, e: IOException) { completer.setException(e) } override fun onResponse(call: Call, response: Response) { ++successes if (successes == 100) { completer.set(Result.success()) } } } completer.addCancellationListener(cancelDownloadsRunnable, executor) repeat(100) { downloadAsynchronously("https://example.com", callback) } callback } } }
Ява
public class CallbackWorker extends ListenableWorker { public CallbackWorker(Context context, WorkerParameters params) { super(context, params); } @NonNull @Override public ListenableFuture<Result> startWork() { return CallbackToFutureAdapter.getFuture(completer -> { Callback callback = new Callback() { int successes = 0; @Override public void onFailure(Call call, IOException e) { completer.setException(e); } @Override public void onResponse(Call call, Response response) { ++successes; if (successes == 100) { completer.set(Result.success()); } } }; completer.addCancellationListener(cancelDownloadsRunnable, executor); for (int i = 0; i < 100; ++i) { downloadAsynchronously("https://www.example.com", callback); } return callback; }); } }
Запуск ListenableWorker в другом процессе
Вы также можете привязать работника к определенному процессу с помощью RemoteListenableWorker
, реализации ListenableWorker
.
RemoteListenableWorker
привязывается к конкретному процессу с помощью двух дополнительных аргументов, которые вы предоставляете как часть входных данных при создании рабочего запроса: ARGUMENT_CLASS_NAME
и ARGUMENT_PACKAGE_NAME
.
В следующем примере показано создание рабочего запроса, привязанного к определенному процессу:
Котлин
val PACKAGE_NAME = "com.example.background.multiprocess" val serviceName = RemoteWorkerService::class.java.name val componentName = ComponentName(PACKAGE_NAME, serviceName) val data: Data = Data.Builder() .putString(ARGUMENT_PACKAGE_NAME, componentName.packageName) .putString(ARGUMENT_CLASS_NAME, componentName.className) .build() return OneTimeWorkRequest.Builder(ExampleRemoteListenableWorker::class.java) .setInputData(data) .build()
Ява
String PACKAGE_NAME = "com.example.background.multiprocess"; String serviceName = RemoteWorkerService.class.getName(); ComponentName componentName = new ComponentName(PACKAGE_NAME, serviceName); Data data = new Data.Builder() .putString(ARGUMENT_PACKAGE_NAME, componentName.getPackageName()) .putString(ARGUMENT_CLASS_NAME, componentName.getClassName()) .build(); return new OneTimeWorkRequest.Builder(ExampleRemoteListenableWorker.class) .setInputData(data) .build();
Для каждого RemoteWorkerService
вам также необходимо добавить определение службы в файл AndroidManifest.xml
:
<manifest ... > <service android:name="androidx.work.multiprocess.RemoteWorkerService" android:exported="false" android:process=":worker1" /> <service android:name=".RemoteWorkerService2" android:exported="false" android:process=":worker2" /> ... </manifest>