理解RxJava:(一)基礎(chǔ)知識(shí)

理解RxJava:(一)基礎(chǔ)知識(shí)

本文翻譯自Grokking RxJava, Part 1: The Basics,著作權(quán)歸原作者danlew所有。譯文由JohnTsai翻譯旭贬。轉(zhuǎn)載請(qǐng)注明出處,并保留此段聲明锐涯。

RxJava這些天成為了Android開(kāi)發(fā)者關(guān)注的新熱點(diǎn)。唯一的問(wèn)題是它在你剛接觸時(shí)難以理解填物。當(dāng)你習(xí)慣了命令式編程全庸,函數(shù)響應(yīng)式編程就變得難以理解秀仲。但是一旦你理解了它融痛,它就變得很棒了壶笼。

我在這試著給你們帶來(lái)不一樣的RxJava。這一系列四篇文章的目標(biāo)是帶你們?nèi)腴T(mén)雁刷。我不會(huì)也不能講解所有的東西覆劈。我只是想讓你們對(duì)RxJava以及它的工作原理感興趣。

基礎(chǔ)知識(shí)

響應(yīng)式代碼的基本構(gòu)成部分是ObservablesSubscribers(譯者注:技術(shù)名詞很難找到合適的中文翻譯沛励,所以維持原文不被翻譯)责语。Observable發(fā)出items,Subscriber消費(fèi)這些items目派。

items如何被消費(fèi)有一套規(guī)則坤候。Observable發(fā)出任意數(shù)量的items(包括0個(gè)items),要么以成功完成終止企蹭,要么以發(fā)生錯(cuò)誤終止白筹。對(duì)于Observable的每個(gè)Subscriber,Observable調(diào)用Subscriber.onNext()方法任意次,然后調(diào)用Subscriber.onComplete()方法或Subscriber.onError()方法谅摄。

這看起來(lái)和我們用的觀察者模式類(lèi)似徒河,但在一個(gè)關(guān)鍵地方不同——Observables在有人明確地訂閱它之后才會(huì)開(kāi)始發(fā)出items。換句話說(shuō)送漠,沒(méi)有人去訂閱顽照,就不會(huì)發(fā)出訂閱事件(譯者注:引申自If a tree falls in a forest)。

Hello World

讓我們通過(guò)一個(gè)具體例子來(lái)看RxJava是如何運(yùn)作的闽寡。首先代兵,先創(chuàng)建一個(gè)基本的Observable:

Observable<String> myObservable = Observable.create(
    new Observable.OnSubscribe<String>() {
        @Override
        public void call(Subscriber<? super String> sub) {
            sub.onNext("Hello, world!");
            sub.onCompleted();
        }
    }
);

Observable發(fā)出Hello World然后完成。現(xiàn)在創(chuàng)建一個(gè)Subscriber來(lái)消費(fèi)掉數(shù)據(jù)爷狈。

Subscriber<String> mySubscriber = new Subscriber<String>() {
    @Override
    public void onNext(String s) { System.out.println(s); }

    @Override
    public void onCompleted() { }

    @Override
    public void onError(Throwable e) { }
};

所有這些所做的是將Observable發(fā)出的每個(gè)String打印出來(lái)植影。

現(xiàn)在有了myObservablemySubscriber,我們可以用subscribe()方法將它們連接起來(lái)。

myObservable.subscribe(mySubscriber);
// 輸出 "Hello, world!"

當(dāng)訂閱發(fā)生時(shí)淆院,myobservable調(diào)用subsriberonNext()onComplete()方法何乎。作為結(jié)果,mySubscriber輸出"Hello,World"然后結(jié)束土辩。

更簡(jiǎn)潔的代碼

為了輸出"Hello,World!",上面寫(xiě)了許多樣板代碼支救。這是因?yàn)槲覟榱俗屇銈兡軌蛎鞔_發(fā)生了什么,選擇了一種啰嗦的方式拷淘。RxJava提供了許多快捷寫(xiě)法讓我們能寫(xiě)出更簡(jiǎn)潔的代碼各墨。

首先,簡(jiǎn)化Observable启涯。RxJava有針對(duì)通用任務(wù)的多種內(nèi)置Observable構(gòu)建方法贬堵。在這種情況下恃轩,Observable.just()發(fā)出一個(gè)item然后完成結(jié)束,就像我們上面的代碼:

Observable<String> myObservable =
    Observable.just("Hello, world!");

然后黎做,對(duì)于啰嗦的Subscriber叉跛。我們不關(guān)心onCompleted()onError()方法,取而代之蒸殿,我們可以用一個(gè)更簡(jiǎn)潔的類(lèi)來(lái)定義在onNext()中做什么:

Action1<String> onNextAction = new Action1<String>() {
    @Override
    public void call(String s) {
        System.out.println(s);
    }
};

Subscriber每部分的Action都能自定義筷厘。Observable.subscribe()能處理一個(gè),兩個(gè)以及三個(gè)Action參數(shù)宏所,以取代onNext(),onError()onComplete()方法酥艳。復(fù)用我們之前的Subscriber,如下:

myObservable.subscribe(onNextAction, onErrorAction, onCompleteAction);

然而,我們僅僅需要第一個(gè)參數(shù)爬骤,因?yàn)槲覀兛梢允÷?code>onError()和onComplete()方法:

myObservable.subscribe(onNextAction);
// 輸出 "Hello, world!"

現(xiàn)在充石,讓我們通過(guò)方法的鏈?zhǔn)秸{(diào)用來(lái)取代這些變量:

Observable.just("Hello, world!")
    .subscribe(new Action1<String>() {
        @Override
        public void call(String s) {
              System.out.println(s);
        }
    });

最后,用Java 8的lambdas表達(dá)式來(lái)去掉丑陋的Action1代碼霞玄。

Observable.just("Hello, world!")
    .subscribe(s -> System.out.println(s));

如果你在Android中使用(迄今為止不能使用Java8)(譯者注:原文作者寫(xiě)這篇文章的時(shí)候(2014年)Java8在Android中開(kāi)發(fā)不能使用骤铃,在譯者翻譯這篇文章的時(shí)候(2016年),已經(jīng)能使用部分特性了)溃列,我推薦使用retrolambda,它將大幅降低代碼的啰嗦程度劲厌。

變換

讓我們把事情變得更有趣。

假設(shè)我想要在輸出的"Hello,world!"語(yǔ)句中加上我的簽名听隐。一種可能(的實(shí)現(xiàn)方式)是改變Observable:

Observable.just("Hello, world! -Dan")
    .subscribe(s -> System.out.println(s));

如果你能夠控制你的Observable,這有效补鼻。但是不能保證以后都是這種情況。如果你使用的是別人的庫(kù)呢雅任?
另一種可能的問(wèn)題是:如果我在多個(gè)地方使用我的Observable风范,但僅僅是某些情況下想要加上簽名呢?

那修改我們的Subscriber怎樣:

Observable.just("Hello, world!")
    .subscribe(s -> System.out.println(s + " -Dan"));

這個(gè)回答同樣不能讓人滿意,有不同的原因:我想要我的Subscriber盡可能輕量沪么,因?yàn)槲铱赡軙?huì)在主線程上運(yùn)行它們硼婿。在更概念的層次上理解,Subscribers被認(rèn)定是做出反應(yīng)(reacts)的事物禽车,而不是做出轉(zhuǎn)變(mutates)的事物寇漫。

如果我能夠通過(guò)一些中間步驟將"Hello,world!"轉(zhuǎn)換,是不是很酷殉摔?

Operators介紹

接下來(lái)是如何解決item轉(zhuǎn)換問(wèn)題:使用operators州胳。Operators被用于在源Observable和最終的Subscriber之間操作被發(fā)出的items。RxJava推出了非常多的operators逸月,但是剛開(kāi)始我們僅僅需要關(guān)注少數(shù)幾個(gè)栓撞。

對(duì)于這種情況,map()操作能被用于將一個(gè)被發(fā)出的item轉(zhuǎn)化為另一個(gè):

Observable.just("Hello, world!")
    .map(new Func1<String, String>() {
        @Override
        public String call(String s) {
            return s + " -Dan";
        }
    })
    .subscribe(s -> System.out.println(s));

同樣的,我們能使用lambda來(lái)簡(jiǎn)化這個(gè):

Observable.just("Hello, world!")
    .map(s -> s + " -Dan")
    .subscribe(s -> System.out.println(s));

非橙肯妫酷瓢颅,我們的map()操作是一個(gè)轉(zhuǎn)換一個(gè)item的Observable。我們可以鏈?zhǔn)秸{(diào)用任意個(gè)map()
弛说,將數(shù)據(jù)改進(jìn)挽懦,成為最終的Subscriber可消費(fèi)的形式。

深入map()

map()有一個(gè)有趣的方面:它不需要發(fā)出和源Observable相同類(lèi)型的items剃浇!

假設(shè)我的Subscriber對(duì)輸出原文本不感興趣巾兆,想要輸出原文本的hash碼:

Observable.just("Hello, world!")
    .map(new Func1<String, Integer>() {
        @Override
        public Integer call(String s) {
            return s.hashCode();
        }
    })
    .subscribe(i -> System.out.println(Integer.toString(i)));

非常有趣——我們以String開(kāi)始但是我們的Subscriber接收的是一個(gè)Integer。

同樣地虎囚,我們能使用lambda來(lái)簡(jiǎn)化代碼:

Observable.just("Hello, world!")
    .map(s -> s.hashCode())
    .subscribe(i -> System.out.println(Integer.toString(i)));

就像我之前說(shuō)的,我們想要Subscriber盡可能少做事蔫磨。通過(guò)另一個(gè)map()來(lái)將hash碼轉(zhuǎn)化為String:

Observable.just("Hello, world!")
    .map(s -> s.hashCode())
    .map(i -> Integer.toString(i))
    .subscribe(s -> System.out.println(s));

你有沒(méi)有發(fā)現(xiàn)——ObservableSubscriber回到了它們之前的樣子了淘讥!我們僅僅在它們之間增加了一些轉(zhuǎn)換步驟。甚至能夠添加我的簽名:

Observable.just("Hello, world!")
   .map(s -> s + " -Dan")
   .map(s -> s.hashCode())
   .map(i -> Integer.toString(i))
   .subscribe(s -> System.out.println(s));

So What堤如?

此刻你可能會(huì)想"對(duì)于一些簡(jiǎn)單的代碼蒲列,用了很多花式步伐一樣技巧"。對(duì)搀罢,那是簡(jiǎn)單的例子蝗岖。但是有兩點(diǎn)你需要掌握:

關(guān)鍵點(diǎn)1:ObservableSubscriber能做任何事情

Observale可以是數(shù)據(jù)庫(kù)查詢,Subscriber得到結(jié)果并將它們顯示在屏幕上榔至。Observable可以是屏幕上的點(diǎn)擊抵赢,Subscriber對(duì)它做出反應(yīng)。Observable可以是從網(wǎng)絡(luò)讀取的字節(jié)流唧取,Subscriber把它寫(xiě)入磁盤(pán)铅鲤。

RxJava是個(gè)能夠處理任何問(wèn)題的通用框架。

關(guān)鍵點(diǎn)2:ObservableSubscriber獨(dú)立于在它們之間的轉(zhuǎn)換步驟

我可以調(diào)用任意次的map操作枫弟,在最初的源Observable和它最終的Subscriber之間邢享。RxJava高度組件化:易于操作數(shù)據(jù)。只要操作于正確的輸入輸出數(shù)據(jù)淡诗,我可以制造一條無(wú)止盡的方法鏈骇塘。

綜合以上兩點(diǎn),我們可以看到RxJava的巨大潛力韩容。雖然此時(shí)我們僅僅有一個(gè)map()操作,這嚴(yán)重地限制了我們的能力款违。在第二部分,我們將深入研究更多RxJava的操作宙攻。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末奠货,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子座掘,更是在濱河造成了極大的恐慌递惋,老刑警劉巖柔滔,帶你破解...
    沈念sama閱讀 216,651評(píng)論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異萍虽,居然都是意外死亡睛廊,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,468評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門(mén)杉编,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)超全,“玉大人,你說(shuō)我怎么就攤上這事邓馒∷恢欤” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 162,931評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵光酣,是天一觀的道長(zhǎng)疏遏。 經(jīng)常有香客問(wèn)我,道長(zhǎng)救军,這世上最難降的妖魔是什么财异? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,218評(píng)論 1 292
  • 正文 為了忘掉前任,我火速辦了婚禮唱遭,結(jié)果婚禮上戳寸,老公的妹妹穿的比我還像新娘。我一直安慰自己拷泽,他們只是感情好疫鹊,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,234評(píng)論 6 388
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著跌穗,像睡著了一般订晌。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上蚌吸,一...
    開(kāi)封第一講書(shū)人閱讀 51,198評(píng)論 1 299
  • 那天锈拨,我揣著相機(jī)與錄音,去河邊找鬼羹唠。 笑死奕枢,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的佩微。 我是一名探鬼主播缝彬,決...
    沈念sama閱讀 40,084評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼哺眯!你這毒婦竟也來(lái)了谷浅?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 38,926評(píng)論 0 274
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎一疯,沒(méi)想到半個(gè)月后撼玄,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,341評(píng)論 1 311
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡墩邀,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,563評(píng)論 2 333
  • 正文 我和宋清朗相戀三年掌猛,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片眉睹。...
    茶點(diǎn)故事閱讀 39,731評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡荔茬,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出竹海,到底是詐尸還是另有隱情慕蔚,我是刑警寧澤,帶...
    沈念sama閱讀 35,430評(píng)論 5 343
  • 正文 年R本政府宣布站削,位于F島的核電站坊萝,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏许起。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,036評(píng)論 3 326
  • 文/蒙蒙 一菩鲜、第九天 我趴在偏房一處隱蔽的房頂上張望园细。 院中可真熱鬧,春花似錦接校、人聲如沸猛频。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,676評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)鹿寻。三九已至,卻和暖如春诽凌,著一層夾襖步出監(jiān)牢的瞬間毡熏,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,829評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工侣诵, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留廷雅,地道東北人摆昧。 一個(gè)月前我還...
    沈念sama閱讀 47,743評(píng)論 2 368
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親赶袄。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,629評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容