ListenableWorker의 스레딩

특정 상황에서는 맞춤설정 스레딩 전략을 제공해야 합니다. 대상 콜백 기반의 비동기 작업을 처리해야 할 수 있습니다. WorkManager는 다음을 사용하여 이 사용 사례를 지원합니다. ListenableWorker ListenableWorker는 가장 기본적인 작업자 API입니다. Worker, CoroutineWorkerRxWorker는 모두 이 클래스에서 파생됩니다. 가 ListenableWorker에서는 직장을 시작하고 중단해야 할 때와 출발해야 할 때만 신호를 보냅니다. 대화목록은 전적으로 본인에게 달려 있습니다. 작업 시작 신호는 기본 따라서 앱의 백그라운드 스레드로 이동하여 선택할 수 있습니다

추상 메서드 ListenableWorker.startWork() 드림 ListenableFuture Result입니다. 가 ListenableFuture는 가벼운 인터페이스로, 다음을 제공하는 Future입니다. 기능을 제공합니다. startWork 메서드를 호출하면 ListenableFuture을 반환해야 합니다. 완료되면 작업의 Result로 설정됩니다. 다음을 만들 수 있습니다. ListenableFuture 인스턴스를 다음 두 가지 방법 중 하나로 사용할 수 있습니다.

  1. Guava를 사용한다면 ListeningExecutorService를 사용합니다.
  2. 그렇지 않은 경우 councurrent-futures 드림 gradle 파일에 추가하고 CallbackToFutureAdapter

비동기 콜백을 기반으로 일부 작업을 실행하려면 다음과 같이 하세요.

Kotlin

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;
        });
    }
}

작업 내용이 중단된 이유는 무엇인가요? 작업이 다음 상태일 때 ListenableWorkerListenableFuture는 항상 취소됩니다. 중단될 것으로 예상됩니다 CallbackToFutureAdapter를 사용하면 취소 리스너를 다음과 같이 호출합니다.

Kotlin

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 실행

ListenableWorker의 구현인 RemoteListenableWorker를 사용하여 작업자를 특정 프로세스에 바인딩할 수도 있습니다.

RemoteListenableWorker는 작업 요청을 작성할 때 입력 데이터의 일부로 제공하는 두 개의 추가 인수 ARGUMENT_CLASS_NAMEARGUMENT_PACKAGE_NAME을 사용하여 특정 프로세스에 바인딩합니다.

다음 예에서는 특정 프로세스에 바인딩된 작업 요청을 빌드하는 방법을 보여줍니다.

Kotlin

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>

샘플