我們?yōu)槭裁匆贏ndroid中使用RxJava

歡迎訪問微涼一季

本文翻譯來自-->Why should we use RxJava on Android

感覺RxJava最近風(fēng)生水起,不學(xué)習(xí)一下都不好意思了,灑家也是初學(xué)RxJava,也是感覺代碼好像更復(fù)雜更難懂了,看了一篇外文感同身受,簡單翻譯一下。本文簡單介紹使用RxJava優(yōu)勢所在。但可能需要有一點(diǎn)RxJava基礎(chǔ),推薦先看一下拋物線的那篇經(jīng)典的匠心寫作。
-----華麗分割線,譯文開始---------



Reactive Extensions (Rx) 是一系列接口和方法,為開發(fā)者提供了一種易懂且迅速簡單易維護(hù)的方法。RxJava就是干這事兒的,提供一系列tools來幫你寫出簡潔的代碼。
老實(shí)說,一開始我認(rèn)為RxJava 寫的代碼理解起來很困難,并且引入一個(gè)庫,單單就是為了用用這種新式的api,這困擾到了我。后來,我懂了。以傳統(tǒng)的編碼方式,隨著app的發(fā)展,我需要重構(gòu)代碼、一遍一遍的重復(fù)樣板代碼,以滿足用戶不斷變更的新需求,這讓我苦不堪言。
我做的大量工作,其實(shí)是改寫相關(guān)方法和接口,就是因?yàn)樾枨蟮淖兏ㄟ@是開發(fā)與產(chǎn)品間那些血案的原罪)或者需要改變展示的信息亦或是需要改變處理信息數(shù)據(jù)..這很抓狂。另外,這種代碼讓其他來維護(hù)的人來理解,通常是很耗時(shí)的。
舉個(gè)栗子:我們需要從數(shù)據(jù)庫獲取一組用戶的鏈表數(shù)據(jù),并展示出來。我們可以用AsyncTask后臺查詢數(shù)據(jù)庫,獲得的結(jié)果給Ui的適配器展示出來。簡單示例代碼:

public class SampleTask extends AsyncTask<Void,Void,List<Users>> {
    private final SampleAdapter mAdapter;

    public SampleTask(SampleAdapter sampleAdapter) {
        mAdapter = sampleAdapater;
    }

    @Override
    protected List<Users> doInBackground(Void... voids) {
        //fetch there results from the database and return them to the onPostExecute
        List<Users> users = getUsersFromDatabase();
        return users;
    }

    @Override
    protected void onPostExecute(List<Users> users) {
        super.onPostExecute(products);
        // Checking if there are users on the database
        if(users == null) {
            //No users, presenting a view saying there are no users
            showEmptyUsersMessageView();
            return;
        }
        for(User user : users){
            mAdapter.add(user);
        }
        mAdapter.notifyDataSetChanged();
    }
}

現(xiàn)在有個(gè)新需求,要求只顯示非guest的user,我們處理的方法是,在添加到adapter前加個(gè)條件判斷是不是guset,或者改變數(shù)據(jù)庫查詢的條件。更有甚者,你又被要求從數(shù)據(jù)庫中獲取另外的其他信息,跟user一并在這個(gè)adapter中顯示出來呢?
這就是我們?yōu)槭裁匆肦xJava了,把我們從這個(gè)泥潭中拉出來。換個(gè)姿勢,我們Rx代碼是這樣子(假設(shè)您已學(xué)習(xí)過Rx基礎(chǔ)用法):

public Observable<List<User>> fetchUsersFromDatabase() {
    return Observable.create(new Observable.OnSubscribe<List<User>(){
        @Override
        public void call(Subscriber<? super List<User>> subscriber){
            // Fetch information from database
            subscriber.onNext(getUserList());
            subscriber.onCompleted();
        }
    });
}

像這樣被調(diào)用:

fetchUsersFromDatabase()
.subscribeOn(Schedulers.io())
//will process everything in a new thread
.observeOn(AndroidSchedulers.mainThread())
//will listen the results on the main thread
.subscribe(new Subscriber<List<User>>() {
           @Override
            public void onCompleted() {
            }
            @Override
            public void onError(Throwable e) {
            }
            @Override
            public void onNext(List<User> users) {
                //Do whatever you want with each user
            }
        });

開始改需求了哈
怎么不顯示guests呢,RxJava分分鐘過濾掉這種不速之客:

fetchUsersFromDatabase()
        .filter(new Func1<User, Boolean>() {
            @Override
            public Boolean call(User user) {
                //only return the users which are not guests
                return !user.isGuest();
            }
        })
        .subscribeOn(Schedulers.io())
        .observeOn(AndroidSchedulers.mainThread())
        .subscribe(new Subscriber<User>() {
                       @Override
                       public void onCompleted() {
                       }

                       @Override
                       public void onError(Throwable e) {
                           /*Check if there was any error while retrieving from database*/
                       }

                       @Override
                       public void onNext(User user) {
                           //Do whatever you want with each user
                       }
                   }
        );

傳統(tǒng)的方式,即便是個(gè)簡單的變更,為了保持優(yōu)雅的接口化編程,我們也得創(chuàng)建新接口,重構(gòu)代碼來實(shí)現(xiàn)過濾。但是使用RxJava讓這一切變得優(yōu)雅了,我們只需要一個(gè)被觀察者用來獲取所有的信息,讓后你就可以盡情的用這些方法來過濾獲取你想要的數(shù)據(jù)。
可能你又會(huì)說了,ok,這是很好很易讀的結(jié)構(gòu),但是這似乎使代碼量變多了呢。well you are right,但是這就是Retrolambda閃耀的時(shí)候了,這個(gè)庫為我們兼容了以使用java8 lambda表達(dá)式,方法引用等等。
幫我們簡化代碼如下:

fetchUsersFromDatabase()
        .subscribeOn(Schedulers.io())
        .observeOn(AndroidSchedulers.mainThread())
        .subscribe(value -> {
                    //Do whatever with the value
                },error -> {
                    //do something with in case of error
                }
        );

這個(gè)問題完美搞定,然后你又開始問了,我需要增加另外的查詢結(jié)果和user一同顯示在這個(gè)adapter中怎么破。這真不是事兒:

fetchUsersFromDatabase()
        .zipWith(fetchSomethingElseFromDatabase(), (users, somethingElse) -> {
            /*here combine users and something else into a new object*/
        })
        .subscribe( o -> {
            /*use the combine object from users and something else to fill the adapter */
});

如上,我們可以輕松組合數(shù)據(jù)庫查出來的其他數(shù)據(jù)和users給一個(gè)adapter一同顯示。是不是更易維護(hù),代碼少,易讀,清晰?
如果要更深入的學(xué)習(xí)RXJava可以看下面這篇文章,我看后受益匪淺。
Party tricks with RxJava, RxAndroid & Retrolambda
另外,這篇教程 tutorial 也幫我在RxJava路上進(jìn)階了很多。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 178,777評論 25 709
  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn),斷路器,智...
    卡卡羅2017閱讀 136,506評論 19 139
  • 前言我從去年開始使用 RxJava ,到現(xiàn)在一年多了。今年加入了 Flipboard 后,看到 Flipboard...
    占導(dǎo)zqq閱讀 9,299評論 6 151
  • 有一天,早上剛來到辦公室,就發(fā)現(xiàn)手機(jī)微信里,有個(gè)好姐妹D給我的截圖,說是她經(jīng)理住的房間隔斷被拆了,我很好奇,19大...
    3e81098bf2ef閱讀 628評論 0 1
  • 根據(jù)架構(gòu)圖進(jìn)行代碼的構(gòu)建。根據(jù)微服務(wù)化設(shè)計(jì)思想,結(jié)合spring cloud一些優(yōu)秀的項(xiàng)目,如服務(wù)發(fā)現(xiàn)、治理、配置...
    swiftie10閱讀 404評論 1 2

友情鏈接更多精彩內(nèi)容