色哟哟视频在线观看-色哟哟视频在线-色哟哟欧美15最新在线-色哟哟免费在线观看-国产l精品国产亚洲区在线观看-国产l精品国产亚洲区久久

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫(xiě)文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

響應(yīng)式編程在Android開(kāi)發(fā)中的實(shí)踐

谷歌開(kāi)發(fā)者 ? 來(lái)源:谷歌開(kāi)發(fā)者 ? 作者:谷歌開(kāi)發(fā)者 ? 2022-04-02 14:07 ? 次閱讀

Flow 是一種基于流的編程模型,本文我們將向大家介紹響應(yīng)式編程以及其在 Android 開(kāi)發(fā)中的實(shí)踐,您將了解到如何將生命周期、旋轉(zhuǎn)及切換到后臺(tái)等狀態(tài)綁定到 Flow 中,并且測(cè)試它們是否能按照預(yù)期執(zhí)行。

單向數(shù)據(jù)流

94d77770-b247-11ec-aa7f-dac502259ad0.png

加載數(shù)據(jù)流的過(guò)程

每款 Android 應(yīng)用都需要以某種方式收發(fā)數(shù)據(jù),比如從數(shù)據(jù)庫(kù)獲取用戶名、從服務(wù)器加載文檔,以及對(duì)用戶進(jìn)行身份驗(yàn)證等。接下來(lái),我們將介紹如何將數(shù)據(jù)加載到 Flow,然后經(jīng)過(guò)轉(zhuǎn)換后暴露給視圖進(jìn)行展示。

為了大家更方便地理解 Flow,我們以 Pancho (潘喬) 的故事來(lái)展開(kāi)。當(dāng)住在山上的 Pancho 想從湖中獲取淡水時(shí),會(huì)像大多數(shù)新手一開(kāi)始一樣,拿個(gè)水桶走到湖邊取水,然后再走回來(lái)。

94ebed04-b247-11ec-aa7f-dac502259ad0.png
山上的 Pancho

但有時(shí) Pahcho 不走運(yùn),走到湖邊時(shí)發(fā)現(xiàn)湖水已經(jīng)干涸,于是就不得不再去別處尋找水源。發(fā)生了幾次這種情況后,Pancho 意識(shí)到,搭建一些基礎(chǔ)設(shè)施可以解決這個(gè)問(wèn)題。于是他在湖邊安裝了一些管道,當(dāng)湖中有水時(shí),只用擰開(kāi)水龍頭就能取到水。知道了如何安裝管道,就能很自然地想到從多個(gè)水源地把管道組合,這樣一來(lái) Pancho 就不必再檢查湖水是否已經(jīng)干涸。

950ccd1c-b247-11ec-aa7f-dac502259ad0.png

鋪設(shè)管道

在 Android 應(yīng)用中您可以簡(jiǎn)單地在每次需要時(shí)請(qǐng)求數(shù)據(jù),例如我們可以使用掛起函數(shù)來(lái)實(shí)現(xiàn)在每次視圖啟動(dòng)時(shí)向 ViewModel 請(qǐng)求數(shù)據(jù),而后 ViewModel 又向數(shù)據(jù)層請(qǐng)求數(shù)據(jù),接下來(lái)這一切又在相反的方向上發(fā)生。不過(guò)這樣過(guò)了一段時(shí)間之后,像 Pancho 這樣的開(kāi)發(fā)者們往往會(huì)想到,其實(shí)有必要投入一些成本來(lái)構(gòu)建一些基礎(chǔ)設(shè)施,我們就可以不再請(qǐng)求數(shù)據(jù)而改為觀察數(shù)據(jù)。觀察數(shù)據(jù)就像安裝取水管道一樣,部署完成后對(duì)數(shù)據(jù)源的任何更新都將自動(dòng)向下流動(dòng)到視圖中,Pancho 再也不用走到湖邊去了。9523d89a-b247-11ec-aa7f-dac502259ad0.png
傳統(tǒng)的請(qǐng)求數(shù)據(jù)與單向數(shù)據(jù)流

響應(yīng)式編程

我們將這類(lèi)觀察者會(huì)自動(dòng)對(duì)被觀察者對(duì)象的變化而作出反應(yīng)的系統(tǒng)稱之為響應(yīng)式編程,它的另一個(gè)設(shè)計(jì)要點(diǎn)是保持?jǐn)?shù)據(jù)只在一個(gè)方向上流動(dòng),因?yàn)檫@樣更容易管理且不易出錯(cuò)。 某個(gè)示例應(yīng)用界面的 "數(shù)據(jù)流動(dòng)" 如下圖所示,身份認(rèn)證管理器會(huì)告訴數(shù)據(jù)庫(kù)用戶已登錄,而數(shù)據(jù)庫(kù)又必須告訴遠(yuǎn)程數(shù)據(jù)源來(lái)加載一組不同的數(shù)據(jù);與此同時(shí)這些操作在獲取新數(shù)據(jù)時(shí)都會(huì)告訴視圖顯示一個(gè)轉(zhuǎn)圈的加載圖標(biāo)。對(duì)此我想說(shuō)這雖然是可行的,但容易出現(xiàn)錯(cuò)誤。
95394c20-b247-11ec-aa7f-dac502259ad0.png
錯(cuò)綜復(fù)雜的 "數(shù)據(jù)流動(dòng)"

更好的方式則是讓數(shù)據(jù)只在一個(gè)方向上流動(dòng),并創(chuàng)建一些基礎(chǔ)設(shè)施 (像 Pancho 鋪設(shè)管道那樣) 來(lái)組合和轉(zhuǎn)換這些數(shù)據(jù)流,這些管道可以隨著狀態(tài)的變化而修改,比如在用戶退出登錄時(shí)重新安裝管道。

95535e6c-b247-11ec-aa7f-dac502259ad0.png

單向數(shù)據(jù)綁定

使用 Flow

可以想象對(duì)于這些組合和轉(zhuǎn)換來(lái)說(shuō),我們需要一個(gè)成熟的工具來(lái)完成這些操作。在本文中我們將使用 Kotlin Flow 來(lái)實(shí)現(xiàn)。Flow 并不是唯一的數(shù)據(jù)流構(gòu)建器,不過(guò)得益于它是協(xié)程的一部分并且得到了很好的支持。我們剛才一直用作比喻的水流,在協(xié)程庫(kù)里稱之為 Flow 類(lèi)型,我們用泛形 T 來(lái)指代數(shù)據(jù)流承載的用戶數(shù)據(jù)或者頁(yè)面狀態(tài)等任何類(lèi)型。

95682fea-b247-11ec-aa7f-dac502259ad0.png

生產(chǎn)者和消費(fèi)者

生產(chǎn)者會(huì)將數(shù)據(jù) emit (發(fā)送) 到數(shù)據(jù)流中,而消費(fèi)者則從數(shù)據(jù)流中 collect (收集) 這些數(shù)據(jù)。在 Android 中數(shù)據(jù)源或存儲(chǔ)區(qū)通常是應(yīng)用數(shù)據(jù)的生產(chǎn)者;消費(fèi)者則是視圖,它會(huì)把數(shù)據(jù)顯示在屏幕上。

大多數(shù)情況下您都無(wú)需自行創(chuàng)建數(shù)據(jù)流,因?yàn)閿?shù)據(jù)源中依賴的庫(kù),例如 DataStore、Retrofit、Room 或 WorkManager 等常見(jiàn)的庫(kù)都已經(jīng)與協(xié)程及 Flow 集成在一起了。這些庫(kù)就像是水壩,它們使用 Flow 來(lái)提供數(shù)據(jù),您無(wú)需了解數(shù)據(jù)是如何生成的,只需 "接入管道" 即可。

9583f086-b247-11ec-aa7f-dac502259ad0.png

提供 Flow 支持的庫(kù)

我們來(lái)看一個(gè) Room 的例子。您可以通過(guò)導(dǎo)出指定類(lèi)型的數(shù)據(jù)流來(lái)獲取數(shù)據(jù)庫(kù)中發(fā)生變更的通知。在本例中,Room 庫(kù)是生產(chǎn)者,它會(huì)在每次查詢后發(fā)現(xiàn)有更新時(shí)發(fā)送內(nèi)容。
@DAOinterface CodelabsDAO {     @Query("SELECT * FROM codelabs")    fun getAllCodelabs(): Flow>}
創(chuàng)建 Flow

如果您要自己創(chuàng)建數(shù)據(jù)流,有一些方案可供選擇,比如數(shù)據(jù)流構(gòu)建器。假設(shè)我們處于 UserMessagesDataSource 中,當(dāng)您希望頻繁地在應(yīng)用內(nèi)檢查新消息時(shí),可以將用戶消息暴露為消息列表類(lèi)型的數(shù)據(jù)流。我們使用數(shù)據(jù)流構(gòu)建器來(lái)創(chuàng)建數(shù)據(jù)流,因?yàn)?Flow 是在協(xié)程上下文環(huán)境中運(yùn)行的,它以掛起代碼塊作為參數(shù),這也意味著它能夠調(diào)用掛起函數(shù),我們可以在代碼塊中使用 while(true) 來(lái)循環(huán)執(zhí)行我們的邏輯。

在示例代碼中,我們首先從 API 獲取消息,然后使用 emit 掛起函數(shù)將結(jié)果添加到 Flow 中,這將掛起協(xié)程直到收集器接收到數(shù)據(jù)項(xiàng),最后我們將協(xié)程掛起一段時(shí)間。在 Flow 中,操作會(huì)在同一個(gè)協(xié)程中順序執(zhí)行,使用 while(true) 循環(huán)可以讓 Flow 持續(xù)獲取新消息直到觀察者停止收集數(shù)據(jù)。傳遞給數(shù)據(jù)流構(gòu)建器的掛起代碼塊通常被稱為 "生產(chǎn)者代碼塊"。

class UserMessagesDataSource(    private val messagesApi: MessagesApi,    private val refreshIntervalMs: Long = 5000) {    val latestMessages: Floa> = flow {        white(true) {            val userMessages = messagesApi.fetchLatestMessages()            emit(userMessages) // 將結(jié)果發(fā)送給 Flow            delay(refreshIntervalMs) //  掛起一段時(shí)間        }    }}

轉(zhuǎn)換 Flow

在 Android 中,生產(chǎn)者和消費(fèi)者之間的層可以使用中間運(yùn)算符修改數(shù)據(jù)流來(lái)適應(yīng)下一層的要求。

在本例中,我們將 latestMessages 流作為數(shù)據(jù)流的起點(diǎn),則可以使用 map 運(yùn)算符將數(shù)據(jù)轉(zhuǎn)換為不同的類(lèi)型,例如我們可以使用 map lambda 表達(dá)式將來(lái)自數(shù)據(jù)源的原始消息轉(zhuǎn)換為 MessagesUiModel,這一操作可以更好地抽象當(dāng)前層級(jí),每個(gè)運(yùn)算符都應(yīng)根據(jù)其功能創(chuàng)建一個(gè)新的 Flow 來(lái)發(fā)送數(shù)據(jù)。我們還可以使用 filter 運(yùn)算符過(guò)濾數(shù)據(jù)流來(lái)獲得包含重要通知的數(shù)據(jù)流。而 catch 運(yùn)算符則可以捕獲上游數(shù)據(jù)流中發(fā)生的異常,上游數(shù)據(jù)流是指在生產(chǎn)者代碼塊和當(dāng)前運(yùn)算符之間調(diào)用的運(yùn)算符產(chǎn)生的數(shù)據(jù)流,而在當(dāng)前運(yùn)算符之后生成的數(shù)據(jù)流則被稱為下游數(shù)據(jù)流。catch 運(yùn)算符還可以在有需要的時(shí)候再次拋出異常或者發(fā)送新值,我們?cè)谑纠a中可以看到其在捕獲到 IllegalArgumentExceptions 時(shí)將其重新拋出,并且在發(fā)生其他異常時(shí)發(fā)送一個(gè)空列表:

val importantUserMessages: Flow<MessageUiModel> =     userMessageDataSource.latestMessages        .map { userMessage ->            userMessages.toUiModel()        }        .filter { messageUiModel ->            messagesUiModel.containsImportantNotifications()        }        .catch { e ->            analytics.log("Error loading reserved event")            if (e is IllegalArgumentException) throw e            else emit(emptyList())}
收集 Flow

現(xiàn)在我們已經(jīng)了解過(guò)如何生成和修改數(shù)據(jù)流,接下來(lái)了解一下如何收集數(shù)據(jù)流。收集數(shù)據(jù)流通常發(fā)生在視圖層,因?yàn)檫@是我們想要在屏幕上顯示數(shù)據(jù)的地方。

在本例中,我們希望列表中能夠顯示最新消息以便 Pancho 能夠了解最新動(dòng)態(tài)。我們可以使用終端運(yùn)算符 collect 來(lái)監(jiān)聽(tīng)數(shù)據(jù)流發(fā)送的所有值,collect 接收一個(gè)函數(shù)作為參數(shù),每個(gè)新值都會(huì)調(diào)用該參數(shù),并且由于它是一個(gè)掛起函數(shù),因此需要在協(xié)程中執(zhí)行。

userMessages.collect{messages->    listAdapter.submitList(messages)}
在 Flow 中使用終端運(yùn)算符將按需創(chuàng)建數(shù)據(jù)流并開(kāi)始發(fā)送值,而相反的是中間操作符只是設(shè)置了一個(gè)操作鏈,其會(huì)在數(shù)據(jù)被發(fā)送到數(shù)據(jù)流時(shí)延遲執(zhí)行。每次對(duì) userMessages 調(diào)用 collect 時(shí)都會(huì)創(chuàng)建一個(gè)新的數(shù)據(jù)流,其生產(chǎn)者代碼塊將根據(jù)自己的時(shí)間間隔開(kāi)始刷新來(lái)自 API 的消息。在協(xié)程中我們將這種按需創(chuàng)建并且只有在被觀察時(shí)才會(huì)發(fā)送數(shù)據(jù)的數(shù)據(jù)流稱之為冷流 (Cold Stream)。

在 Android 視圖上收集數(shù)據(jù)流

在 Android 的視圖中收集數(shù)據(jù)流要注意兩點(diǎn),第一是在后臺(tái)運(yùn)行時(shí)不應(yīng)浪費(fèi)資源,第二是配置變更。

安全收集

假設(shè)我們?cè)?MessagesActivity 中,如果希望在屏幕上顯示消息列表,則應(yīng)該當(dāng)界面沒(méi)有顯示在屏幕上時(shí)停止收集,就像是 Pancho 在刷牙或者睡覺(jué)時(shí)應(yīng)該關(guān)上水龍頭一樣。我們有多種具有生命周期感知能力的方案,來(lái)實(shí)現(xiàn)當(dāng)信息不在屏幕上展示就不從數(shù)據(jù)流中收集信息的功能,比如 androidx.lifecycle:lifecycle-runtime-ktx 包中的 Lifecycle.repeatOnLifecycle(state)Flow.flowWithLifecycle(lifecycle, state)。您還可以在 ViewModel 中使用 androidx.lifecycle:lifecycle-livedata-ktx 包里的 Flow.asLiveData(): LiveData 將數(shù)據(jù)流轉(zhuǎn)換為 LiveData,這樣就可以像往常一樣使用 LiveData 來(lái)實(shí)現(xiàn)這件事情。不過(guò)為了簡(jiǎn)單起見(jiàn),這里推薦使用 repeatOnLifecycle 從界面層收集數(shù)據(jù)流。

repeatOnLifecycle 是一個(gè)接收 Lifecycle.State 作為參數(shù)的掛起函數(shù),該 API 具有生命周期感知能力,所以能夠在當(dāng)生命周期進(jìn)入響應(yīng)狀態(tài)時(shí)自動(dòng)使用傳遞給它的代碼塊啟動(dòng)新的協(xié)程,并且在生命周期離開(kāi)該狀態(tài)時(shí)取消該協(xié)程。在上面的例子中,我們使用了 Activity 的 lifecycleScope 來(lái)啟動(dòng)協(xié)程,由于 repeatOnLifecycle 是掛起函數(shù),所以它需要在協(xié)程中被調(diào)用。最佳實(shí)踐是在生命周期初始化時(shí)調(diào)用該函數(shù),就像上面的例子中我們?cè)?Activity 的 onCreate 中調(diào)用一樣:

import androidx.lifecycle.repeatOnLifecycle class MessagesActivity : AppCompatActivity() {     val viewModel: MessagesViewModel by viewModels()     override fun onCreate(savedInstanceState: Bundle?) {                       lifecycleScope.launch {                repeatOnLifecycle(Lifecycle.State.STARTED)                    viewModel.userMessages.collect { messages ->                        listAdapter.submitList(messages)                    }                }                // 協(xié)程將會(huì)在 lifecycle 進(jìn)入 DESTROYED 后被恢復(fù)            }    }}

repeatOnLifecycle 的可重啟行為充分考慮了界面的生命周期,不過(guò)需要注意的是,直到生命周期進(jìn)入 DESTROYED,調(diào)用 repeatOnLifecycle 的協(xié)程都不會(huì)恢復(fù)執(zhí)行,因此如果您需要從多個(gè)數(shù)據(jù)流中進(jìn)行收集,則應(yīng)在 repeatOnLifecycle 代碼塊內(nèi)多次使用 launch 來(lái)創(chuàng)建協(xié)程:

lifecycleScope.launch {    repeatOnLifecycle(Lifecycle.State.STARTED) {             launch {                viewModel.userMessages.collect { … }            }             launch {                otherFlow.collect { … }            }    }}

如果只需從一個(gè)數(shù)據(jù)流中進(jìn)行收集,則可使用 flowWithLifecycle 來(lái)收集數(shù)據(jù),它能夠在生命周期進(jìn)入目標(biāo)狀態(tài)時(shí)發(fā)送數(shù)據(jù),并在離開(kāi)目標(biāo)狀態(tài)時(shí)取消內(nèi)部的生產(chǎn)者:

lifecycleScope.launch {    viewModel.userMessages        .flowWithLifecycle(lifecycle, State.STARTED)        .collect { messages ->            listAdapter.submitList(messages)        }}

為了能夠直觀地展示具體的運(yùn)作過(guò)程,我們來(lái)探索一下此 Activity 的生命周期,首先是創(chuàng)建完成并向用戶可見(jiàn);接下來(lái)用戶按下了主屏幕按鈕將應(yīng)用退到后臺(tái),此時(shí) Activity 會(huì)收到 onStop 信號(hào);當(dāng)重新打開(kāi)應(yīng)用時(shí)又會(huì)調(diào)用 onStart。如果您調(diào)用 repeatOnLifecycle 并傳入 STARTED 狀態(tài),界面就只會(huì)在屏幕上顯示時(shí)收集數(shù)據(jù)流發(fā)出的信號(hào),并且在應(yīng)用轉(zhuǎn)到后臺(tái)時(shí)取消收集。

95951910-b247-11ec-aa7f-dac502259ad0.png

Activity 的生命周期

repeatOnLifecycleflowWithLifecycle 是 lifecycle-runtime-ktx 庫(kù)在 2.4.0 穩(wěn)定版中新增的 API,在沒(méi)有這些 API 之前您可能已經(jīng)以其他方式從 Android 界面中收集數(shù)據(jù)流,例如像上面的代碼一樣直接從 lifecycleScope.launch 啟動(dòng)的協(xié)程中收集,雖然這樣看起來(lái)也能工作但不一定安全,因?yàn)檫@種方式將持續(xù)從數(shù)據(jù)流中收集數(shù)據(jù)并更新界面元素,即便是應(yīng)用退出到后臺(tái)時(shí)也一樣。如果使用 launchWhenStarted 替代它的話,情況會(huì)稍微好一些,因?yàn)樗鼤?huì)在處于后臺(tái)時(shí)將收集掛起。但這樣會(huì)在讓數(shù)據(jù)流生產(chǎn)者保持活躍狀態(tài),有可能會(huì)在后臺(tái)持續(xù)發(fā)出不需要在屏幕上顯示的數(shù)據(jù)項(xiàng),從而將內(nèi)存占滿。由于界面并不知道數(shù)據(jù)流生產(chǎn)者的實(shí)現(xiàn)方式,所以最好謹(jǐn)慎一些,使用 repeatOnLifecycleflowWithLifecycle 來(lái)避免界面在處于后臺(tái)時(shí)收集數(shù)據(jù)或保持?jǐn)?shù)據(jù)流生產(chǎn)者處于活躍狀態(tài)。 下面是一段不安全的使用方式示例:
class MessagesActivity : AppCompatActivity() {     val viewModel: MessagesViewModel by viewModels()     override fun onCreate(savedInstanceState: Bundle?) {             //  危險(xiǎn)的操作            lifecycleScope.launch {                viewModel.userMessage.collect { messages ->                    listAdapter.submitList(messages)                }            }             //  危險(xiǎn)的操作            LifecycleCoroutineScope.launchWhenX {                flow.collect { … }            }    }}

配置變更

當(dāng)您向視圖暴露數(shù)據(jù)流時(shí),必須要考慮到您正在嘗試在具有不同生命周期的兩個(gè)元素之間傳遞數(shù)據(jù),并不是所有生命周期都會(huì)出現(xiàn)問(wèn)題,但在 Activity 和 Fragment 的生命周期里會(huì)比較棘手。當(dāng)設(shè)備旋轉(zhuǎn)或者接收到配置變更時(shí),所有的 Activity 都可能會(huì)重啟但 ViewModel 卻能被保留,因此您不能把任意數(shù)據(jù)流都簡(jiǎn)單地從 ViewModel 中暴露出來(lái)。

95aab50e-b247-11ec-aa7f-dac502259ad0.png

旋轉(zhuǎn)屏幕會(huì)重建 Activity 但能夠保留 ViewModel

以如下代碼中的冷流為例,由于每次收集冷流時(shí)它都會(huì)重啟,所以在設(shè)備旋轉(zhuǎn)之后會(huì)再次調(diào)用 repository.fetchItem()。我們需要某種緩沖區(qū)機(jī)制來(lái)保障無(wú)論重新收集多少次都可以保持?jǐn)?shù)據(jù),并在多個(gè)收集器之間共享數(shù)據(jù),而 StateFlow 正是為了此用途而設(shè)計(jì)的。在我們的湖泊比喻中,StateFlow 就好比水箱,即使沒(méi)有收集器它也能持有數(shù)據(jù)。因?yàn)樗梢远啻伪皇占阅軌蚍判牡貙⑵渑c Activity 或 Fragment 一起使用。

val result: Flow>> = flow {    emit(repository.fetchItem())}

您可以使用 StateFlow 的可變版本,并隨時(shí)根據(jù)需要在協(xié)程中更新它的值,但這樣做可能不太符合響應(yīng)式編程的風(fēng)格,如下代碼所示:
private val _myUiState = MutableStateFlow() val myUiState: StateFlow = _myUiState init {    viewModelScope.launch {        _muUiState.value = Result.Loading        _myUiState.value = repository.fetchStuff()    }}

Pancho 會(huì)建議您將各種類(lèi)型的數(shù)據(jù)流都轉(zhuǎn)換為 StateFlow 來(lái)改進(jìn)這個(gè)問(wèn)題,這樣 StateFlow 將接收來(lái)自上游數(shù)據(jù)流的所有更新并存儲(chǔ)最新的值,并且收集器的數(shù)量可以是 0 至任意多個(gè),因此非常適合與 ViewModel 一起使用。當(dāng)然,除此之外還有一些其他類(lèi)型的 Flow,但推薦您使用 StateFlow,因?yàn)槲覀兛梢詫?duì)它進(jìn)行非常精確的優(yōu)化。

95bf4bcc-b247-11ec-aa7f-dac502259ad0.png

將任意數(shù)據(jù)流轉(zhuǎn)換為 StateFlow

要將數(shù)據(jù)流轉(zhuǎn)換為 StateFlow 可以使用 stateIn 運(yùn)算符,它需要傳入三個(gè)參數(shù): initinalValuescopestarted。其中 initialValue 是因?yàn)?StateFlow 必須有值;而協(xié)程 scope 則是用于控制何時(shí)開(kāi)始共享,在上面的例子中我們使用了 viewModelScope;最后的 started 是個(gè)有趣的參數(shù),我們后面會(huì)聊到 WhileSubscribed(5000) 的作用,先看這部分的代碼:
val result: StateFlow> = someFlow    .stateIn(        initialValue = Result.Loading        scope = viewModelScope,        started = WhileSubscribed(5000),)

我們來(lái)看看這兩個(gè)場(chǎng)景: 第一種場(chǎng)景是旋轉(zhuǎn),在該場(chǎng)景中 Activity (也就是數(shù)據(jù)流收集器) 在短時(shí)間內(nèi)被銷(xiāo)毀然后重建;第二個(gè)場(chǎng)景是回到主屏幕,這將會(huì)使我們的應(yīng)用進(jìn)入后臺(tái)。在旋轉(zhuǎn)場(chǎng)景中我們不希望重啟任何數(shù)據(jù)流以便盡可能快地完成過(guò)渡,而在回到主屏幕的場(chǎng)景中我們則希望停止所有數(shù)據(jù)流以便節(jié)省電量和其他資源。

我們可以通過(guò)設(shè)置超時(shí)時(shí)間來(lái)正確判斷不同的場(chǎng)景,當(dāng)停止收集 StateFlow 時(shí),不會(huì)立即停止所有上游數(shù)據(jù)流,而是會(huì)等待一段時(shí)間,如果在超時(shí)前再次收集數(shù)據(jù)則不會(huì)取消上游數(shù)據(jù)流,這就是 WhileSubscribed(5000) 的作用。當(dāng)設(shè)置了超時(shí)時(shí)間后,如果按下主屏幕按鈕會(huì)讓視圖立即結(jié)束收集,但 StateFlow 會(huì)經(jīng)過(guò)我們?cè)O(shè)置的超時(shí)時(shí)間之后才會(huì)停止其上游數(shù)據(jù)流,如果用戶再次打開(kāi)應(yīng)用則會(huì)自動(dòng)重啟上游數(shù)據(jù)流。而在旋轉(zhuǎn)場(chǎng)景中視圖只停止了很短的時(shí)間,無(wú)論如何都不會(huì)超過(guò) 5 秒鐘,因此 StateFlow 并不會(huì)重啟,所有的上游數(shù)據(jù)流都將會(huì)保持在活躍狀態(tài),就像什么都沒(méi)有發(fā)生一樣可以做到即時(shí)向用戶呈現(xiàn)旋轉(zhuǎn)后的屏幕。

95d9a5f8-b247-11ec-aa7f-dac502259ad0.png

設(shè)置超時(shí)時(shí)間來(lái)應(yīng)對(duì)不同的場(chǎng)景

總的來(lái)說(shuō),建議您使用 StateFlow 來(lái)通過(guò) ViewModel 暴露數(shù)據(jù)流,或者使用 asLiveData 來(lái)實(shí)現(xiàn)同樣的目的,關(guān)于 StateFlow 或其父類(lèi) SharedFlow 的更多詳細(xì)信息,請(qǐng)參閱:StateFlow 和 SharedFlow。

測(cè)試數(shù)據(jù)流

測(cè)試數(shù)據(jù)流可能會(huì)比較復(fù)雜,因?yàn)橐幚淼膶?duì)象是流式數(shù)據(jù),這里介紹在兩個(gè)不同的場(chǎng)景中有用的小技巧: 首先是第一個(gè)場(chǎng)景,被測(cè)單元依賴了數(shù)據(jù)流,那對(duì)此類(lèi)場(chǎng)景進(jìn)行測(cè)試最簡(jiǎn)單的方法就是用模擬生產(chǎn)者替代依賴項(xiàng)。在本例中,您可以對(duì)這個(gè)模擬源進(jìn)行編程以對(duì)不同的測(cè)試用例發(fā)送其所需要的內(nèi)容。您可以像上面的例子一樣實(shí)現(xiàn)一個(gè)簡(jiǎn)單的冷流,測(cè)試本身會(huì)對(duì)受測(cè)對(duì)象的輸出進(jìn)行斷言,輸出的內(nèi)容可以是數(shù)據(jù)流或其他任何類(lèi)型。

95f27f42-b247-11ec-aa7f-dac502259ad0.png

被測(cè)單元依賴數(shù)據(jù)流的測(cè)試技巧

模擬被測(cè)單元所依賴的數(shù)據(jù)流:
class MyFakeRepository : MyRepository {    fun observeCount() = flow {        emit(ITEM_1)    }}

如果受測(cè)單元暴露一個(gè)數(shù)據(jù)流,并且您希望驗(yàn)證該值或一系列值,那么您可以通過(guò)多種方式收集它們。您可以對(duì)數(shù)據(jù)流調(diào)用 first() 方法以進(jìn)行收集并在接收到第一個(gè)數(shù)據(jù)項(xiàng)后停止收集。您還可以調(diào)用 take(5) 并使用 toList 終端操作符來(lái)收集恰好 5 條消息,這種方法可能非常有幫助。

9608e8d6-b247-11ec-aa7f-dac502259ad0.png
測(cè)試數(shù)據(jù)流的技巧

測(cè)試數(shù)據(jù)流:
@Testfun myTest() = runBlocking {     // 收集第一個(gè)數(shù)據(jù)然后停止收集    val firstItem = repository.counter.first()     // 收集恰好 5 條消息    val first = repository.messages.take(5).toList()}
回顧

感謝閱讀本文,希望您通過(guò)本文內(nèi)容已經(jīng)了解到為什么響應(yīng)式架構(gòu)值得投資,以及如何使用 Kotlin Flow 構(gòu)建您的基礎(chǔ)設(shè)施。文末提供了有關(guān)這方面的資料,包括涵蓋基礎(chǔ)知識(shí)的指南以及深入探討某些主題的文章。另外您還可以通過(guò) Google I/O 應(yīng)用了解這些內(nèi)容的詳細(xì)信息,我們?cè)谠缧r(shí)候?yàn)槠涓铝撕芏嘤嘘P(guān)數(shù)據(jù)流的內(nèi)容。

原文標(biāo)題:實(shí)戰(zhàn) | 使用 Kotlin Flow 構(gòu)建數(shù)據(jù)流 "管道"

文章出處:【微信公眾號(hào):谷歌開(kāi)發(fā)者】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

審核編輯:湯梓紅
聲明:本文內(nèi)容及配圖由入駐作者撰寫(xiě)或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問(wèn)題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • Android
    +關(guān)注

    關(guān)注

    12

    文章

    3941

    瀏覽量

    127729
  • 開(kāi)發(fā)
    +關(guān)注

    關(guān)注

    0

    文章

    370

    瀏覽量

    40886
  • 編程
    +關(guān)注

    關(guān)注

    88

    文章

    3637

    瀏覽量

    93901
  • Flow
    +關(guān)注

    關(guān)注

    0

    文章

    10

    瀏覽量

    8856

原文標(biāo)題:實(shí)戰(zhàn) | 使用 Kotlin Flow 構(gòu)建數(shù)據(jù)流 "管道"

文章出處:【微信號(hào):Google_Developers,微信公眾號(hào):谷歌開(kāi)發(fā)者】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦

    前端開(kāi)發(fā)之函數(shù)編程實(shí)踐

    函數(shù)編程是一種編程范式,它將程序抽象為函數(shù)和數(shù)據(jù)結(jié)構(gòu),通過(guò)函數(shù)調(diào)用來(lái)實(shí)現(xiàn)程序的功能,并且函數(shù)可以作為參數(shù)傳遞給其他函數(shù)。 JavaScript
    發(fā)表于 06-05 10:19 ?473次閱讀
    前端<b class='flag-5'>開(kāi)發(fā)</b>之函數(shù)<b class='flag-5'>式</b><b class='flag-5'>編程</b><b class='flag-5'>實(shí)踐</b>

    嵌入式開(kāi)發(fā),你是如何理解嵌入C編程的?

    `本文來(lái)探討嵌入式開(kāi)發(fā)的嵌入C編程,看看跟您理解的一樣嗎?一、新手常常問(wèn)的一個(gè)問(wèn)題:C語(yǔ)言和嵌入C
    發(fā)表于 03-22 09:38

    嵌入工程師系統(tǒng)培訓(xùn)就業(yè)班

    嵌入C語(yǔ)言編程基礎(chǔ)主要介紹嵌入式開(kāi)發(fā)編程C語(yǔ)言的重要概念和
    發(fā)表于 07-24 15:36

    實(shí)踐中學(xué)嵌入linux應(yīng)用程序開(kāi)發(fā)

    編程android應(yīng)用編程android播放器項(xiàng)目設(shè)計(jì)等。重視應(yīng)用是貫穿全書(shū)的最大特點(diǎn),各章和全書(shū)結(jié)尾分別設(shè)置了
    發(fā)表于 12-08 10:49

    《從實(shí)踐中學(xué)習(xí)嵌入Linux操作系統(tǒng)》高清PDF資源分享!

    內(nèi)容簡(jiǎn)介:《從實(shí)踐中學(xué)嵌入linux應(yīng)用程序開(kāi)發(fā)》結(jié)合大量實(shí)例,講解了嵌入linux應(yīng)用程序設(shè)計(jì)各個(gè)方面的基本方法及必要的核心概念。主要內(nèi)容包括搭建嵌入
    發(fā)表于 11-01 16:23

    實(shí)踐中學(xué)嵌入Linux應(yīng)用程序開(kāi)發(fā)

    《從實(shí)踐中學(xué)嵌入Linux應(yīng)用程序開(kāi)發(fā)》內(nèi)容簡(jiǎn)介: 《從實(shí)踐中學(xué)嵌入linux應(yīng)用程序開(kāi)發(fā)
    發(fā)表于 11-04 06:36

    Android開(kāi)發(fā)寶典之Android程序員入門(mén)

    本部分討論,我提到兩種不同的開(kāi)發(fā)者:傳統(tǒng)的桌面應(yīng)用程序開(kāi)發(fā),他們能使用任何的編程語(yǔ)言,而且最終的產(chǎn)品和程序是用來(lái)運(yùn)行桌面操作系統(tǒng)的;還有
    發(fā)表于 09-05 15:16 ?546次下載
    <b class='flag-5'>Android</b><b class='flag-5'>開(kāi)發(fā)</b>寶典之<b class='flag-5'>Android</b>程序員入門(mén)

    Google+Android開(kāi)發(fā)實(shí)踐代碼大全

    Android 是用于移動(dòng)設(shè)備的軟件堆棧,包括操作系統(tǒng)、中間件和關(guān)鍵應(yīng)用程序。Android SDK 提供了必需的工具和 API,用于開(kāi)始開(kāi)發(fā)Android 驅(qū)動(dòng)的設(shè)備上運(yùn)行的應(yīng)用
    發(fā)表于 09-07 11:11 ?366次下載
    Google+<b class='flag-5'>Android</b><b class='flag-5'>開(kāi)發(fā)</b>與<b class='flag-5'>實(shí)踐</b>代碼大全

    Android開(kāi)發(fā)的經(jīng)驗(yàn)教學(xué)

    本文主要用來(lái)收集Android開(kāi)發(fā)積累的一些寶貴經(jīng)驗(yàn),這些經(jīng)驗(yàn)中有一些約定熟成且經(jīng)過(guò)檢驗(yàn)的建議,有一些結(jié)合最新技術(shù)的實(shí)踐。無(wú)論是菜鳥(niǎo)還是大神,都應(yīng)該學(xué)會(huì)閱讀別人的經(jīng)驗(yàn),并結(jié)合自己的思
    發(fā)表于 10-10 10:47 ?0次下載
    <b class='flag-5'>Android</b><b class='flag-5'>開(kāi)發(fā)</b><b class='flag-5'>中</b>的經(jīng)驗(yàn)教學(xué)

    ARM嵌入系統(tǒng)開(kāi)發(fā)_Android應(yīng)用開(kāi)發(fā)入門(mén)(基礎(chǔ)版)

    ARM嵌入系統(tǒng)開(kāi)發(fā)_Android應(yīng)用開(kāi)發(fā)入門(mén)(基礎(chǔ)版)(嵌入式開(kāi)發(fā)板教程)-該文檔為ARM嵌入
    發(fā)表于 08-04 11:02 ?0次下載
    ARM嵌入<b class='flag-5'>式</b>系統(tǒng)<b class='flag-5'>開(kāi)發(fā)</b>_<b class='flag-5'>Android</b>應(yīng)用<b class='flag-5'>開(kāi)發(fā)</b>入門(mén)(基礎(chǔ)版)

    基于DM6646開(kāi)發(fā)板的多線程編程嵌入圖像處理的應(yīng)用

    基于DM6646開(kāi)發(fā)板的多線程編程嵌入圖像處理的應(yīng)用(嵌入式開(kāi)發(fā)北京)-文檔為基于DM66
    發(fā)表于 08-04 15:29 ?10次下載
    基于DM6646<b class='flag-5'>開(kāi)發(fā)</b>板的多線程<b class='flag-5'>編程</b><b class='flag-5'>在</b>嵌入<b class='flag-5'>式</b>圖像處理<b class='flag-5'>中</b>的應(yīng)用

    《從實(shí)踐中學(xué)嵌入Linux應(yīng)用程序開(kāi)發(fā)》pdf完整版資源分享

    《從實(shí)踐中學(xué)嵌入Linux應(yīng)用程序開(kāi)發(fā)》內(nèi)容簡(jiǎn)介: 《從實(shí)踐中學(xué)嵌入linux應(yīng)用程序開(kāi)發(fā)
    發(fā)表于 11-01 17:07 ?0次下載
    《從<b class='flag-5'>實(shí)踐</b>中學(xué)嵌入<b class='flag-5'>式</b>Linux應(yīng)用程序<b class='flag-5'>開(kāi)發(fā)</b>》pdf完整版資源分享

    第二章 Android系統(tǒng)與嵌入式開(kāi)發(fā)

    第二章Android系統(tǒng)與嵌入式開(kāi)發(fā)第二章首先要先了解Android和嵌入Lnux系統(tǒng)有什么區(qū)別和聯(lián)系,嵌入Linux系統(tǒng)是
    發(fā)表于 11-02 20:51 ?13次下載
    第二章 <b class='flag-5'>Android</b>系統(tǒng)與嵌入<b class='flag-5'>式開(kāi)發(fā)</b>

    Android操作系統(tǒng)移植到嵌入平臺(tái)的最佳實(shí)踐

      遵循上述嵌入工程實(shí)踐,您可以確保嵌入平臺(tái)上高效成功地移植 Android,并增強(qiáng)現(xiàn)有連接設(shè)備的功能或從頭開(kāi)始設(shè)計(jì)新系統(tǒng),而無(wú)需太多
    的頭像 發(fā)表于 07-01 10:20 ?2048次閱讀

    基于Android開(kāi)發(fā)的ADT獲取內(nèi)存的敏感信息

    Eclipse上的開(kāi)發(fā)工具。目前Android開(kāi)發(fā)所用的開(kāi)發(fā)工具主要有Android Studio 、intellij idea、Eclip
    的頭像 發(fā)表于 09-06 15:22 ?1220次閱讀
    主站蜘蛛池模板: 美女夫妻内射潮视频| 亚洲幼女网| 色狠狠色综合吹潮| 手机观看毛片| 亚洲高清在线mv| 中文字幕无线手机在线| 白百合在线观看| 国产精品免费视频播放| 近亲乱中文字幕| 蜜芽国产在线精品欧美| 三级黄色网| 中文字幕人成乱码熟女APP| 国产精品18久久久久久欧美| 久操久操久操| 欧美日韩亚洲成人| 小776论坛| 18禁三级黄| 耻辱の奴隷淑女中文字幕| 干性感美女| 精品国产乱码久久久久乱码 | 两性午夜色视频免费网站| 全免费午夜一级毛片| 午夜剧场1000| 最近中文字幕MV免费高清视频8| 成人在免费视频手机观看网站| 久久婷婷久久一区二区三区| 日韩精品真人荷官无码| 亚洲色噜噜狠狠网站| V8成品人视频| 好紧小嫩嫩水的10p| 欧美精品AV精品一区视频| 一道本在线伊人蕉无码| 99热精品一区| 国语对白嫖老妇胖老太| 亲爱的妈妈6韩国电影免费观看| 亚洲欧美在无码片一区二区| 国产3级在线观看| 久久综合伊人 magnet| 睡觉被偷偷进入magnet| videossex性暴力| 美女漏bb|