摘要:來吧,我們看看不用單個線程池,如何實現(xiàn)隊列。也就是說,只有在上傳成功時,繼續(xù)執(zhí)行下一個。最終執(zhí)行的如下圖好了,混亂的總算擼成串了。
前言
前幾天和Stay(有心課堂創(chuàng)始人)討論了一個rxjava場景,覺得很有意思,這里也和大家分享下。通過還原一個真實的開發(fā)過程,來感受下rxjava的便利之處。
不認(rèn)識Stay的可以看看他的博客吧:http://www.jianshu.com/users/...
巨坑從來都是由小坑慢慢塌陷的先來看下一段最普通的代碼
在沒有特殊需求的情況下,代碼就這么簡單。你可以理解為,獲取一個目錄下的所有文件,將它們一個個傳到服務(wù)器上去。
看起來好像是沒什么問題,一個for循環(huán)搞定。一個task失敗了不影響另一個task。每個task run在一個多帶帶的子線程。
之前rxjava使用場景只局限于和Retrofit一起用。沒過多的使用操作符。因此在uploadFile(path)方法中就是最簡單的Retrofit+Rxjava上傳文件。rxjava就切換了下線程。
對于寫慣java的人,這么寫是沒什么問題的。但如果深入使用過rxjava之后,這么寫就非常別扭了??吹絝or loop了,你不想將它改成Observable.from()嘛?
把能看見的都改成stream吧getFileList()方法是獲取sd卡中data包下所有以loc為后綴的files。
workflow分三步:
locate to data dir
list files under data dir
filter files with .loc suffix
換成rxjava非常容易
先發(fā)射一個data目錄路徑
需求是多次上傳文件,得用flatMap將data映射成一個Observable
2.1 當(dāng)然你可以選擇直接listFile(filter),但這樣回調(diào)又套回調(diào),不是很好看。 2.2 用filter操作符將發(fā)射來的File[]過濾
比如像2.1這樣寫
或者像2.2這樣寫
注意,在flatMap中又用from()操作符將File[]變換成一個個Observable
假如你的API接口可以接收多個文件,其實也不用這樣寫。直接在flatMap中拼接RequestBody,調(diào)用API請求就可以了。比如像下圖這樣寫:
![rx04(http://img.blog.csdn.net/2016...
無奈需求是上傳loc文件同時還會再帶上一個sensor文件,所以就不能像上述這樣寫。
接下來的workflow就很有趣了?,F(xiàn)在有了多個Observable
如果不考慮隊列,不考慮無網(wǎng)或上傳失敗情況。完全再來一個flatMap將Observable
但現(xiàn)在的需求是,隊列上傳文件,也就是說,必須一個任務(wù)完成(成功|失敗)后才能進(jìn)行下一個任務(wù)。這樣用flatMap就不可以了。(其實后來我考慮過這個問題,線程的調(diào)度本質(zhì)還是由我聲明出來的線程池來決定的,如果用Schedulers.newThread(),那就會創(chuàng)建多個子線程。但如果用Schedulers.from(Executors.newSingleThreadExecutor())呢?)
需求總是多變的,好在有rxjava可以隨意變換。來吧,我們看看不用單個線程池,如何實現(xiàn)隊列。
不能隨意套路,坑的是自己之前學(xué)習(xí)rxjava時,看過很多在android中高度使用rxjava的文章。有一個操作符很有意思-> concat()
The Concat operator concatenates the output of multiple Observables so that they act like a single Observable, with all of the items emitted by the first Observable being emitted before any of the items emitted by the second Observable (and so forth, if there are more than two).
即將多個Observables串起成一個Observable,直到一個執(zhí)行完畢后再執(zhí)行下一個。
我們可以將這個concat()應(yīng)用在讀取緩存還是請求服務(wù)器, 如果緩存有數(shù)據(jù),那就不用請求服務(wù)器了。
Observable cache; Observable server; Observable.concat(cache, server) .first()
這個也可以用在隊列上傳文件場景上咯。but,concat()是創(chuàng)建型操作符,再次變換就不能使用了。不過可以用concatMap(),
Returns a new Observable that emits items resulting from applying a function that you supply to each item emitted by the source Observable, where that function returns an Observable, and then emitting the items that result from concatenating those resulting Observables.
直接看代碼吧
這段寫的特別扭,為什么又要在一個Observable里又創(chuàng)建一個retrofit相關(guān)的Observable?當(dāng)時想的是,因為要在upload成功后得刪除文件啊。如果把subscribe放到外層去,那接收到的全是服務(wù)器response,不知道當(dāng)前的response屬于哪一個file upload。所以我就又寫了次變換。(這里肯定可以優(yōu)化的,寫的太挫)
在concatMap中接收到from()發(fā)射來的一個Observable,變換成Retrofit請求,當(dāng)Subscriber標(biāo)記為onCompleted后再去執(zhí)行下一個Observable。
到這里還沒完,假如無網(wǎng)絡(luò)又或者服務(wù)器異常。在第一個Observable就會失敗,此時還需要繼續(xù)請求嗎?很有可能后面的Observable也都不成功。那加個判斷吧。concat()可以和first()一起用。concatMap()也是可以的。
If you are only interested in the first item emitted by an Observable, or the first item that meets some criteria, you can filter the Observable with the First operator.
如果first() -> return true; 這樣只取到目前的這個Observable,后續(xù)的不執(zhí)行了。
也就是說,只有在上傳成功時return false,繼續(xù)執(zhí)行下一個Observable。否則就return true停止。
覺醒分割線我想之前肯定是被concat(cache, db, server).first()整懵逼了,一心去套,才寫了上面這么二的代碼的。等等,容我換個姿勢。
看,對請求結(jié)果map變換一次就可以啦,如果成功刪除相關(guān)文件,不成功就是個異常了。Observable.error()。這樣就跳出了concatMap,也就是說,當(dāng)異常發(fā)生時會停止后續(xù)的文件上傳。這樣first()也不需要啦。除非還有其他額外的停止flag要判斷。
到這里整個workflow就被rxjava梳理完畢了。是不是很有趣?我們來看下代碼全過程。
還剩最后一個問題:線程調(diào)度。
之前一直都沒寫線程調(diào)度的地方。subscribeOn放在哪里比較好?
需求是:在主線程listFile拿到目錄下的所有文件,然后在子線程一個個隊列上傳文件,執(zhí)行完畢后再切換到主線程彈dialog告知結(jié)果。
這樣來說,每個文件上傳時不需要切換線程,所以調(diào)用retrofit的地方是不需要subscribeOn。如果執(zhí)意要在uploadTrip()后加上subscribeOn(io),也不是不可以。只是每個上傳task都在一個新的線程里執(zhí)行的。但實際上,我們的文件上傳是個隊列,完全可以一直在同一個線程里執(zhí)行。所以我放在了flatMap中。最終執(zhí)行的log如下圖
好了,混亂的workflow總算擼成串了。平時看相關(guān)文章總覺得很簡單,無非就是幾個操作符拼接在一起,做了線程切換。不好理解的就是鏈?zhǔn)剿季S的轉(zhuǎn)換還有一些操作符:compose transformer等。等到真正應(yīng)用到項目場景中,著實折騰了不少。比如不用flatMap,改為concatMap。比如線程調(diào)度。比如放棄使用retrofit+rxjava套路,重新認(rèn)識reactive等。
總得來說,當(dāng)理解了rxjava的鏈?zhǔn)剿季S并對一些復(fù)雜的邏輯重構(gòu)之后,還是會愛上的。
參考閱讀
理解操作符,還是看官網(wǎng)最佳ReactiveX
感謝小鄧子幫忙梳理流程。
有心課堂,傳遞給你的不僅僅是技術(shù)!如果你有任何問題,請加入我們的QQ討論群:172448270
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/65045.html
摘要:開發(fā)這樣一款加載器最大原因,就是為了代碼分離以及靜態(tài)資源模塊化無縫接合。異步方式,將依賴分割成多個節(jié)點,然后每個節(jié)點形成一個新的文件塊。會處理最常見的模塊標(biāo)準(zhǔn)和。通過聲明依賴,可以選擇一個本地的版本,而不需要使用全局聲明的版本。 作者:Jogis 原文鏈接:https://github.com/yesvods/Blog/issues/2 轉(zhuǎn)載請注明原文鏈接以及作者信息 模塊加載器...
摘要:開發(fā)這樣一款加載器最大原因,就是為了代碼分離以及靜態(tài)資源模塊化無縫接合。異步方式,將依賴分割成多個節(jié)點,然后每個節(jié)點形成一個新的文件塊。會處理最常見的模塊標(biāo)準(zhǔn)和。通過聲明依賴,可以選擇一個本地的版本,而不需要使用全局聲明的版本。 作者:Jogis 原文鏈接:https://github.com/yesvods/Blog/issues/2 轉(zhuǎn)載請注明原文鏈接以及作者信息 模塊加載器...
閱讀 3992·2021-11-22 15:31
閱讀 2525·2021-11-18 13:20
閱讀 3115·2021-11-15 11:37
閱讀 7043·2021-09-22 15:59
閱讀 748·2021-09-13 10:27
閱讀 3783·2021-09-09 09:33
閱讀 1448·2019-08-30 15:53
閱讀 2570·2019-08-29 15:37