Architecture -- WorkManager

1. WorkManager

1). 簡介

其實就是"管理一些要在后臺工作的任務, -- 即使你的應用沒啟動也能保證任務能被執行",WorkManager在底層, 會根據你的設備情況, 選用JobScheduler, Firebase的JobDispatcher, 或是AlarmManager。WorkManager并不是為了那種在應用內的后臺線程而設計出來的. 這種需求你應該使用ThreadPool。

2). 引入
  • kolin
# Android Studio 3.0, 3.0以下版本請使用complie
  ext.work_version = "1.0.0-alpha02"
  // WorkManager
  implementation "android.arch.work:work-runtime:$work_version" // use -ktx for Kotlin
  // optional - Firebase JobDispatcher support
  implementation "android.arch.work:work-firebase:$work_version"
  // optional - Test helpers
  androidTestImplementation "android.arch.work:work-testing:$work_version"
  • java
# 必須加入
implementation "android.arch.work:work-runtime:1.0.0-alpha01"

2. 用法

1). 執行一次的任務
  • PullWorker.kt
/**
 * Pull 任務
 * 創建步驟:
 *      1. 繼承Worker類
 *      2. 重寫doWork()方法
 * Created by mazaiting on 2018/6/6.
 */
class PullWorker : Worker() {
    override fun doWork(): WorkerResult {
        // 模擬設置頁面中的"是否接受推送"是否被勾選
        val isOkay = this.inputData.getBoolean("key_accept_bg_work", false)

        // 判斷是否推送
        if (isOkay) {
            // 模擬長時間工作
            Thread.sleep(5000)
            // 獲取數據
            val pulledResult = startPull()
            // 設置輸出數據
            val output = Data.Builder().putString("key_pulled_result", pulledResult).build()
            // 存放數據
            outputData = output
            // 返回成功
            return WorkerResult.SUCCESS
        } else {
            // 返回失敗
            return WorkerResult.FAILURE
        }
    }

    /**
     * 開始推送
     */
    private fun startPull(): String {
        return "[worker] pull messages from backend."
    }

}
  • PullEngine.kt
/**
 * Pull 引擎
 * Created by mazaiting on 2018/6/6.
 */
class PullEngine {

    fun scheduleOneTimeWork(context: Context) {
        // 設置任務,執行一次
        val pullRequest = OneTimeWorkRequestBuilder<PullWorker>()
                .setInputData(
                        Data.Builder()
                                .putBoolean("key_accept_bg_work", true)
                                .build()
                )
                .build()
        WorkManager.getInstance().enqueue(pullRequest)
        // 將請求入列
        WorkManager.getInstance().enqueue(pullRequest)
        // 獲取請求ID
        val pullRequestId = pullRequest.id
        // 存在SharedPreference中
        PreferenceManager
                .getDefaultSharedPreferences(context)
                .edit()
                .putString("pullId", pullRequestId.toString())
                .apply()
    }

}
  • MainApplication.kt(注意在AndroidManifest.xml文件中配置)
class MainApplication : Application() {
    override fun onCreate() {
        super.onCreate()

        // 任務執行一次
        PullEngine().scheduleOneTimeWork(this.applicationContext)
    }
}
  • MainActivity.kt
class MainActivity : AppCompatActivity() {

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)
        // UUID實現了Serializable接口,能由toString(),fromString()與String互轉
        val uuid = UUID.fromString(
                PreferenceManager.getDefaultSharedPreferences(this.applicationContext)
                        .getString("pullId", "")
        )
        WorkManager.getInstance()
                // 獲取狀態ID
                .getStatusById(uuid)
                // 觀察數據
                .observe(this, Observer<WorkStatus>{ status ->
                    if (null != status && status.state.isFinished) {
                        val pulledResult = status.outputData.getString("key_pulled_result","")
                        Log.e("MainActivity", pulledResult)
                    }
                })
    }
}
  • 打印結果


    圖1.png
2). 任務設置約束
    fun scheduleOneTimeWork(context: Context) {
        val constraints = Constraints.Builder()
                // 負荷
                .setRequiresCharging(true)
                .build()
        // 設置任務,執行一次
        val pullRequest = OneTimeWorkRequestBuilder<PullWorker>()
                // 設置約束
                .setConstraints(constraints)
                // 輸入數據
                .setInputData(
                        Data.Builder()
                                .putBoolean("key_accept_bg_work", true)
                                .build()
                )
                .build()
        // 將請求入列
        WorkManager.getInstance().enqueue(pullRequest)
        // 獲取請求ID
        val pullRequestId = pullRequest.id
        // 存在SharedPreference中
        PreferenceManager
                .getDefaultSharedPreferences(context)
                .edit()
                .putString("pullId", pullRequestId.toString())
                .apply()
    }
3). 取消任務
    // 獲取請求ID
    val pullRequestId = pullRequest.id
    // 取消任務
    WorkManager.getInstance().cancelWorkById(pullRequestId)
4). 任務鏈
WorkManager.getInstance()
    .beginWith(workA)
        // Note: WorkManager.beginWith() returns a
        // WorkContinuation object; the following calls are
        // to WorkContinuation methods
    .then(workB)    // FYI, then() returns a new WorkContinuation instance
    .then(workC)
    .enqueue()

WorkManager.getInstance()
    // First, run all the A tasks (in parallel):
    .beginWith(workA1, workA2, workA3)
    // ...when all A tasks are finished, run the single B task:
    .then(workB)
    // ...then run the C tasks (in any order):
    .then(workC1, workC2)
    .enqueue()

再或

val chain1 = WorkManager.getInstance()
    .beginWith(workA)
    .then(workB)
val chain2 = WorkManager.getInstance()
    .beginWith(workC)
    .then(workD)
val chain3 = WorkContinuation
    .combine(chain1, chain2)
    .then(workE)
chain3.enqueue()
5). 定時任務
    /**
     * 執行任務
     */
    fun schedulePeriodicWork(context: Context) {
        val constraints = Constraints.Builder()
                // 負荷
                .setRequiresCharging(true)
                .build()
        // 設置任務執行時長,并設置輸入參數
        // 長期任務,循環
        val pullRequest = PeriodicWorkRequestBuilder<PullWorker>(30, TimeUnit.SECONDS)
                            .setConstraints(constraints)
                            .setInputData(
                                Data.Builder()
                                        .putBoolean("key_accept_bg_work", true)
                                        .build()
                            )
                            .build()
        // 將請求入列
        WorkManager.getInstance().enqueue(pullRequest)
        // 獲取請求ID
        val pullRequestId = pullRequest.id
        // 存在SharedPreference中
        PreferenceManager
                .getDefaultSharedPreferences(context)
                .edit()
                .putString("pullId", pullRequestId.toString())
                .apply()
    }
6). 原文地址
7). 代碼下載
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 227,572評論 6 531
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,071評論 3 414
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 175,409評論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,569評論 1 307
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,360評論 6 404
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 54,895評論 1 321
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 42,979評論 3 440
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,123評論 0 286
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,643評論 1 333
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,559評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,742評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,250評論 5 356
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 43,981評論 3 346
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,363評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,622評論 1 280
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,354評論 3 390
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,707評論 2 370