Skip to content

Latest commit

 

History

History
198 lines (160 loc) · 6.83 KB

README.md

File metadata and controls

198 lines (160 loc) · 6.83 KB

WorkManager

WorkManager는 상황별 실행과 보장된 실행을 조합하여 적용해야 하는 백그라운드 작업을 위한 아키텍처 구성요소이다.

  • 상황별 실행이란 최대한 빨리 백그라운드 작업을 실행하는 것
  • 보장된 실행이란 사용자가 앱을 벗어난 경우를 비롯한 다양한 상황에서 로직을 처리하는 것

WorkManager 종속 항목

WorkManager는 다음과 같은 종속 항목이 필요하다.

dependencies {
    // WorkManager dependency
    implementation("androidx.work:work-runtime-ktx:2.8.1")  
}

BlurWorker 클래스 생성

  1. worker는 백그라운드 스레드에서 동기식으로 작업을 실행하는 클래스이다.
    비동기식 작업을 하기 위해선 Kotlin의 코루틴과 상호 운용되는 CoroutineWorker를 사용할 수 있다. CoroutineWorker안에는 doWork()메서드가 필수적으로 재정의되어야 한다.
class BlurWorker(context: Context, params: WorkerParameters) : CouroutineWorker(context, params) {
    override suspend fun doWork(): Result { ... }
}
  1. WorkManager는 Result.success()Result.failure()를 사용하여 실행 중인 작업 요청의 최종 상태를 나타낸다. 따라서 return try ..catch 코드 블록을 추가한다.
//inside doWork()
return try {
    Result.success()
} catch(throwable: Throwable) {
    Result.failure()
}
  1. 변환하고자 하는 이미지를 비트맵으로 변환하여 변수에 저장한다. 이후에 호출되는 blurBitmap()은 비트맵을 블러 처리하는 메서드이고, writeBitmapToFile()은 비트앱을 시스템에 저장하고 Uri를 반환하는 메서드이다.
//inside try block
val picture = BitmapFactory.decodeResource(
                applicationContext.resources,
                R.drawable.android_cupcake
)

//output's type: Bitmap
val output = blurBitmap(picture, 1)
val outputUri = writeBitmapToFile(applicationContext, output)

Result.Success()
  1. catch 블록엔 에러 메세지를 기록한다.
//inside doWork()
} catch (throwable: Throwable) {
    Log.e(
    TAG,
    applicationContext.resources.getString(R.string.error_applying_blur),
        throwable
    )

    Result.failure()
}
  1. CoroutineWorker는 기본적으로 Dispatchers.Default로 실행된다. withContext()를 호출하고 원하는 디스패처를 전달하여 변경할 수 있다.
//inside doWork(), wrapping try-catch block
return withContext(Dispatchers.IO) {
    ...
}

6. 람다 함수 내에서 return을 호출할 수 없어서 `try-catch` 블록의 return에서 에러가 발생한다. 이러한 경우 상태를 직접 표시하는 라벨을 부착한다.
```kotlin
return withContext(Dispatchers.IO) {
    return@withContext try{
        ...
    } catch ...
}
  1. worker는 매우 빨리 실행된다. 따라서 약간의 지연을 추가해서 작업의 속도를 늦추도록 한다.
return withContext(Dispatchers.IO) {
    delay(DELAY_TIME_MILLIS)
    ...
}

WorkRequest 추가

WorkRequest에는 2가지 유형이 있다. 이 예제에선 첫 번째 유형을 사용한다.

  • OneTimeWorkRequest: 한 번만 실행된다.
  • PeriodicWorkRequest: 일정 주기로 반복적으로 실행된다.
  1. 저장소 클래스 안에 OneTimeWorkerRequestBuilder라는 확장 함수를 사용해 worker용 OneTimeWorkRequest 객체를 만든다.
//WorkManagerBluromaticRepository.kt
fun applyBlur(blurLevel: Int){
    val blurBuilder = OneTimeWorkRequestBuilder<BlurWorker>()
}
  1. workManager 객체의 enqueue 메서드를 통해 WorkRequest를 실행 큐에 넣는다.
val workManager = WorkManager.getInstance(context)

fun applyBlur(blurLevel: Int){
    val blurBuilder = OneTimeWorkRequestBuilder<BlurWorker>()
    
    workManager.enqueue(blurBuilder.build())
}

입력 데이터 객체 만들기

Worker의 안팎에서 Data 객체를 통해 입력과 출력을 전달할 수 있다. Data는 키-값 쌍의 가벼운 컨테이너로 소량의 데이터를 저장하기에 안성맞춤이다.

  1. Data 객체를 생성하고 반환하는 함수를 선언한다.
private fun createInputDataForWorkRequest(blurLevel: Int, imageUri: Uri): Data {
    val builder = Data.Builder()
    builder.putString(KEY_IMAGE_URI, imageUri.toString()).putInt(BLUR_LEVEL, blurLevel)
    return builder.build()
}
  1. WorkRequestBuildersetInputData()메서드를 통해 생성한 Data 객체를 넘겨준다.
override fun applyBlur(blurLevel: Int) {
     // Create WorkRequest to blur the image
    val blurBuilder = OneTimeWorkRequestBuilder<BlurWorker>()

    // New code for input data object
    blurBuilder.setInputData(createInputDataForWorkRequest(blurLevel, imageUri))

    workManager.enqueue(blurBuilder.build())
} 
  1. Worker 객체에서 입력 데이터에 접근하려면 inputDatagetString(), getInt()와 같은 메서드를 사용한다.
override fun doWork(): Result {
    val resourceUri = inputData.getString(KEY_IMAGE_URI)
    val blurLevel = inputData.getInt(KEY_BLUR_LEVEL, 1)
    ...
}
  1. 전달받은 데이터의 존재 여부를 확인하고 데이터를 사용하도록 한다. require문은 조건을 충족하지 않으면 블록 내 코드를 실행한다.
return@withContext try {
    // NEW code
    require(!resourceUri.isNullOrBlank()) {
        val errorMessage =
            applicationContext.resources.getString(R.string.invalid_input_uri)
            Log.e(TAG, errorMessage)
            errorMessage
    }

출력 데이터 객체 만들기

출력 데이터도 동일하게 Data 객체를 생성하여 만들어낸다.

  1. Result.success()의 인자로 Data 객체를 넘겨서 출력으로 내보낼 수 있다. workDataOf로 키-값 쌍의 데이터 객체를 만들 수 있다.
val outputData = workDataOf(KEY_IMAGE_URI to outputUri.toString())
Result.success(outputData)

작업 체이닝

WorkManager는 작업 체인 기능을 지원한다. 이는 순서대로 실행되거나 동시에 실행되는 별도의 WorkerRequest를 만들 수 있다.

  1. WorkManagerbeginWith() 메서드를 호출하여 제일 처음 실행할 WorkRequest를 지정한다. beginWith()WorkContinuation 객체를 반환하고 체인의 시작점이 된다.
override fun applyBlur(blurLevel: Int) {
    var continuation = workManager.beginWith(OneTimeWorkRequest.from(CleanupWorker::class.java))
    ...
}       
  1. then() 메서드를 호출하고 WorkRequest객체를 전달하여 이 작업 요청 체인에 추가할 수 있다.
val blurBuilder = OneTimeWorkRequestBuilder<BlurWorker>()
continuation = continuation.then(blurBuilder.build())
  1. 작업을 시작하려면 enqueue() 메서드를 호출한다.
continuation.enqueue()