利用Angular2的Observables實(shí)現(xiàn)交互控制的方法
在Angular1.x中,我們使用Promise來處理各種異步。但是在angular2中,使用的是Reactive Extensions (Rx)的Observable。對于Promise和Observable的區(qū)別,網(wǎng)上有很多文章,推薦egghead.io上的這個(gè)7分鐘的視頻(作者 Ben Lesh)。在這個(gè)視頻的介紹中,主要說的,使用Observable創(chuàng)建的異步任務(wù),可以被處理,而且是延時(shí)加載的。這篇文章里,我們主要針對一些在跟服務(wù)器端交互的時(shí)候遇到的問題,來看看Observable給我們帶來的特性。
實(shí)例場景
首先,我們來定義一下問題的場景。假設(shè)我們要實(shí)現(xiàn)一個(gè)搜索功能,有一個(gè)簡單的輸入框,當(dāng)用戶輸入文字的時(shí)候,實(shí)時(shí)的利用輸入的文字進(jìn)行查詢,并顯示查詢的結(jié)果。
問題
在這個(gè)簡單的場景當(dāng)中,一般需要考慮3個(gè)問題:
不能在用戶輸入每個(gè)字符的時(shí)候就觸發(fā)搜索。
如果用戶輸入每個(gè)字符就觸發(fā)搜索,一來浪費(fèi)服務(wù)器資源,二來客戶端頻繁觸發(fā)搜索,以及更新搜索結(jié)果,也會(huì)影響客戶端的響應(yīng)。一般這個(gè)問題,都是通過加一些延時(shí)來避免。
如果用戶輸入的文本沒有變化,就不應(yīng)該重新搜索。
假設(shè)用戶輸入了'foo'以后,停頓了一會(huì),觸發(fā)了搜索,再敲了一個(gè)字符'o',結(jié)果發(fā)現(xiàn)打錯(cuò)了,又刪掉了這個(gè)字符。如果這個(gè)時(shí)候用戶又停頓一會(huì),導(dǎo)致觸發(fā)了搜索,這次的文本'foo'跟之前搜索的時(shí)候的文本是一樣的,所以不應(yīng)該再次搜索。
要考慮服務(wù)器的異步返回的問題。
當(dāng)我們使用異步的方式往服務(wù)器端發(fā)送多個(gè)請求的時(shí)候,我們需要注意接受返回的順序是無法保證的。比如我們先后搜索了2個(gè)單詞'computer', ‘car', 雖然'car'這個(gè)詞是后來搜的,但是有可能服務(wù)器處理這個(gè)搜索比較快,就先返回結(jié)果。這樣頁面就會(huì)先顯示'car'的搜索結(jié)果,然后等收到'computer'的搜索結(jié)果的時(shí)候,再顯示'computer'的結(jié)果。但是,這時(shí)候在用戶看來明明搜索的是'car',卻顯示的是另外的結(jié)果。
迎接挑戰(zhàn)
在這個(gè)實(shí)例中,我們使用wikipedia的api接口來開發(fā)一個(gè)簡單的實(shí)例,實(shí)現(xiàn)簡單的搜索功能。
實(shí)現(xiàn)搜索
由于只是演示,我們的app里面只包含2個(gè)文件: app.ts 和 wikipedia-service.ts,最終版本的源文件,請參考原文提供的demo鏈接。
我們直接來看最初版本的WikipediaService是如何實(shí)現(xiàn)的:
import { Injectable } from '@angular/core'; import { URLSearchParams, Jsonp } from '@angular/http'; @Injectable() export class WikipediaService { constructor(private jsonp: Jsonp) {} search (term: string) { var search = new URLSearchParams() search.set('action', 'opensearch'); search.set('search', term); search.set('format', 'json'); return this.jsonp .get('http://en.wikipedia.org/w/api.php?callback=JSONP_CALLBACK', { search }) .toPromise() .then((response) => response.json()[1]); } }
在這里版本中,使用Jsonp模塊來請求api結(jié)果,它的結(jié)果應(yīng)該是一個(gè)類型為Observable<Response>的對象,我們把返回的結(jié)果從Observable<Response> 轉(zhuǎn)換成 Promise<Response>對象,然后使用它的then方法把結(jié)果轉(zhuǎn)成json。這樣,這個(gè)search方法的返回類型為Promise<Array<string>>。
注意上面,我們使用response.json()[1]方式,從原先的結(jié)果中,得到我們需要的查詢結(jié)果的列表,列表里面都是string。
這個(gè)看起來很簡單,在angular1.x里面,也基本都是使用$http或$resource,來返回一個(gè)Promise類型的結(jié)果。
下面就是app.ts的部分內(nèi)容(因?yàn)檫@只是演示,所以直接在app.ts里面直接定義module和component,并且調(diào)用service,在真實(shí)的app中,應(yīng)該創(chuàng)建相應(yīng)的組件來實(shí)現(xiàn)):
// check the plnkr for the full list of imports import {...} from '...'; @Component({ selector: 'my-app', template: ` <div> <h2>Wikipedia Search</h2> <input #term type="text" (keyup)="search(term.value)"> <ul> <li *ngFor="let item of items">{{item}}</li> </ul> </div> ` }) export class AppComponent { items: Array<string>; constructor(private wikipediaService: WikipediaService) {} search(term) { this.wikipediaService.search(term).then(items => this.items = items); } }
從上面的代碼也能看出,AppComponent有一個(gè)search()方法,它調(diào)用wikipediaService.search()方法,因?yàn)檫@個(gè)方法返回一個(gè)Promise<Array<string>>類型的結(jié)果,所以使用then(),把結(jié)果列表賦值給model對象items。上面的template里面的模板內(nèi)容就是用來以列表顯示查詢的結(jié)果。
雖然這個(gè)實(shí)現(xiàn)滿足了基本的查詢功能,但是對于上面提到的3個(gè)問題,都沒有能夠解決。下面就來修改這個(gè)實(shí)現(xiàn)來解決上面的問題。
控制用戶輸入延時(shí)
我們先解決第一個(gè)問題:當(dāng)用戶輸入的時(shí)候,不要每次輸入一個(gè)字符就觸發(fā)一次搜索,而是設(shè)置一個(gè)時(shí)間延時(shí),當(dāng)用戶停止輸入的時(shí)間超過400毫秒,就觸發(fā)搜索。如果用戶一直不停的輸入,輸入的時(shí)間間隔小于400ms就不觸發(fā)。這正是'Observables'能做的事情。
為此,我們需要一個(gè)Observable<string>對象來保存用戶的輸入,然后就可以用這個(gè)對象提供的方法來實(shí)現(xiàn)延時(shí)觸發(fā)的功能。我們可以利用Angular2的指令(directive)formControl。要用這個(gè)指令,需要引入ReactiveFormsModule模塊。
import { NgModule } from '@angular/core'; import { BrowserModule } from '@angular/platform-browser'; import { JsonpModule } from '@angular/http'; import { ReactiveFormsModule } from '@angular/forms'; @NgModule({ imports: [BrowserModule, JsonpModule, ReactiveFormsModule] declarations: [AppComponent], bootstrap: [AppComponent] }) export class AppModule {}
引入以后,我們就可以在模板里面使用FormControl來創(chuàng)建表單輸入,并給他設(shè)置一個(gè)變量名term。
<input type="text" [formControl]="term"/>
這樣,這個(gè)input組件所綁定的變量term就是FormControl的一個(gè)實(shí)例,它有一個(gè)屬性valueChanges,這個(gè)屬性是一個(gè)Observable<string>類型的對象。我們就可以使用Observable<string>的debounceTime方法來設(shè)置觸發(fā)延時(shí)。
export class AppComponent { items: Array<string>; term = new FormControl(); constructor(private wikipediaService: WikipediaService) { this.term.valueChanges .debounceTime(400) .subscribe(term => this.wikipediaService.search(term).then(items => this.items = items)); } }
我們看到this.term.valueChanges是一個(gè)Observable<string>對象,通過debounceTime(400)我們設(shè)置它的事件觸發(fā)延時(shí)是400毫秒。這個(gè)方法還是返回一個(gè)Observable<string>對象。然后我們就給這個(gè)對象添加一個(gè)訂閱事件:
term => this.wikipediaService.search(term).then(items => this.items = items)
這是用lambda表達(dá)式寫的一個(gè)方法。參數(shù)term就是Observable<string>對象經(jīng)過400ms的延時(shí)設(shè)置,產(chǎn)生的一個(gè)用戶輸入的字符串。方法體就是用這個(gè)參數(shù)進(jìn)行搜索,跟之前版本的處理方式一致。
在這個(gè)修改版中,我們把之前的search()方法去掉,直接在構(gòu)造函數(shù)constructor(...)里面添加的,這相當(dāng)于,用戶在輸入框的輸入,是一個(gè)消息源,會(huì)經(jīng)過debounceTime(400)的處理,然后產(chǎn)生一個(gè)消息,這個(gè)消息會(huì)發(fā)送給訂閱的事件處理函數(shù)來處理,也就是搜索。所以,我們不需要一個(gè)search()方法來控制什么時(shí)候觸發(fā),而是通過類型訂閱的機(jī)制來處理用戶輸入。
防止觸發(fā)兩次
現(xiàn)在我們再來解決第二個(gè)問題,就是經(jīng)過400ms的延時(shí)以后,用戶輸入的搜索條件一樣的情況。有了上面的Observable,這個(gè)就很簡單了,Observable有一個(gè)distinctUntilChanged的方法,他會(huì)判斷從消息源過來的新數(shù)據(jù)跟上次的數(shù)據(jù)是否一致,只有不一致才會(huì)觸發(fā)訂閱的方法。
this.term.valueChanges .debounceTime(400) .distinctUntilChanged() .subscribe(term => this.wikipediaService.search(term).then(items => this.items = items));
處理返回順序
上面描述了服務(wù)器端異步返回?cái)?shù)據(jù)的時(shí)候,返回順序不一致出現(xiàn)的問題。對于這個(gè)問題,我們的解決辦法就比較直接,也就是對于之前的請求返回的結(jié)果,直接忽略,只處理在頁面上用戶最后一次發(fā)起的請求的結(jié)果。說道忽略之前的請求,如果你們看了上面的視頻,或者知道Promise和Observable的區(qū)別的話,就應(yīng)該想到我們可以利用Observable的dispose()方法來解決。實(shí)際上,我們是利用這種'disposable'特性來解決,而不是直接調(diào)用dispose()方法。(實(shí)在不知道該怎么翻譯'disposable',它的意思是我可以中止在Observable對象上的消息處理,字面的意思是可被丟棄的、一次性的。)
上面我們講到,在service的search()方法里,我們把Jsonp返回的結(jié)果從Observable<Response> 轉(zhuǎn)換成 Promise<Response>對象。為了利用Observable的特性去丟棄上一個(gè)未及時(shí)返回的結(jié)果,我們讓這個(gè)方法還是返回Observable類型的結(jié)果。下面就是修改后的WikipediaService里面的search()方法。
search (term: string) { var search = new URLSearchParams() search.set('action', 'opensearch'); search.set('search', term); search.set('format', 'json'); return this.jsonp .get('http://en.wikipedia.org/w/api.php?callback=JSONP_CALLBACK', { search }) .map((response) => response.json()[1]); }
注意這個(gè)方法最后用.map((response) => response.json()[1]),意思是對于原先的Response類型的結(jié)果,轉(zhuǎn)換成實(shí)際的搜索結(jié)果的列表。
map()以及后面要說到的flatMap()之類的方法,是函數(shù)式編程里面常用到的方法,意思就是將原先的數(shù)據(jù)集里面的每一條數(shù)據(jù),經(jīng)過一定的處理再返回一個(gè)新的結(jié)果,也就是把一個(gè)數(shù)據(jù)集轉(zhuǎn)換成另一個(gè)數(shù)據(jù)集。
現(xiàn)在,我們的WikipediaSerice的返回結(jié)果就不是Promise了,所以我們就需要修改app.ts,我們不能再使用then()方法來處理結(jié)果,而是使用subscribe()添加一個(gè)消息訂閱方法。
this.term.valueChanges .debounceTime(400) .distinctUntilChanged() .subscribe( term => this.wikipediaService.search(term).subscribe( items => this.items = items ) );
其中,第一個(gè)subscribe():
this.term.valueChanges...subscribe(term => ....)
這個(gè)是對輸入框產(chǎn)生的查詢字符串,注冊一個(gè)訂閱方法,來處理用戶的輸入。
第二個(gè)subscribe():
this.wikipediaService.search(term).subscribe(items => this.items = items));
是對從服務(wù)器端返回的數(shù)據(jù)查詢結(jié)果,注冊一個(gè)訂閱方法,來將這個(gè)數(shù)據(jù)賦值到model上。
我們也可以用下面的方式,來避免這樣使用多個(gè)subscribe:
this.term.valueChanges .debounceTime(400) .distinctUntilChanged() .flatMap(term => this.wikipediaService.search(term)) .subscribe(items => this.items = items);
我們在用戶輸入的字符串的Observable<string>上調(diào)用flatMap(...)方法,相當(dāng)于,對用戶輸入的每個(gè)有效的查詢條件,調(diào)用wikipediaService.search()方法。然后對這個(gè)查詢返回的數(shù)據(jù),再注冊一個(gè)訂閱方法。
費(fèi)了這么大的篇幅,希望你明白了Observable的flatMap和subscribe用法,對于沒有接觸過函數(shù)式編程的人來說,這確實(shí)不好理解,但是在Angular2里面,我們將會(huì)大量使用各種函數(shù)式編程的方法。所以還是需要你花時(shí)間慢慢理解。
費(fèi)了這么大功夫,上面說的似乎跟'忽略之前未及時(shí)返回的消息'好像沒什么關(guān)系,那么上面的修改到底有沒有解決那個(gè)問題呢。沒有!確實(shí)是沒有。因?yàn)槲覀兪褂胒latMap,對用戶輸入的每個(gè)有效的查詢字符串,都會(huì)調(diào)用訂閱的那個(gè)處理函數(shù),然后更新model。所以我們的問題還是沒有解決。
但是到了這一步以后,解決辦法就很容易了,我們只需要用switchMap代理flatMap就可以。就這么簡單!這是因?yàn)椋瑂witchMap會(huì)在處理每一個(gè)新的消息的時(shí)候,就直接把上一個(gè)消息注冊的訂閱方法直接取消掉。
最后,再優(yōu)化一下代碼:
@Component({ selector: 'my-app', template: ` <div> <h2>Wikipedia Search</h2> <input type="text" [formControl]="term"/> <ul> <li *ngFor="let item of items | async">{{item}}</li> </ul> </div> ` }) export class AppComponent { items: Observable<Array<string>>; term = new FormControl(); constructor(private wikipediaService: WikipediaService) { this.items = this.term.valueChanges .debounceTime(400) .distinctUntilChanged() .switchMap(term => this.wikipediaService.search(term)); } }
我們直接把switchMap()的結(jié)果,賦給model對象this.items,也就是一個(gè)Observable<Array<string>>類型的數(shù)據(jù)。這樣,在模板里面使用items的地方也需要修改,使用AsyncPipe就可以:
<li *ngFor="let item of items | async">{{item}}</li>
這樣,模板在解析items這個(gè)model的時(shí)候,就會(huì)自動(dòng)解析這個(gè)Observable的結(jié)果,再渲染頁面。
Demo地址: Smart Wikipedia search using Angular 2
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
angular 表單驗(yàn)證器驗(yàn)證的同時(shí)限制輸入的實(shí)現(xiàn)
表單驗(yàn)證是經(jīng)常用到一個(gè)東西,這篇文章主要介紹了angular 表單驗(yàn)證器驗(yàn)證的同時(shí)限制輸入的實(shí)現(xiàn),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-04-04Angular2中constructor和ngOninit的使用講解
這篇文章主要介紹了Angular2中constructor和ngOninit的使用講解,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-05-05AngularJS學(xué)習(xí)筆記之基本指令(init、repeat)
AngularJS 指令是擴(kuò)展的 HTML 屬性,帶有前綴 ng-。ng-app 指令初始化一個(gè) AngularJS 應(yīng)用程序。ng-init 指令初始化應(yīng)用程序數(shù)據(jù)。ng-model 指令把應(yīng)用程序數(shù)據(jù)綁定到 HTML 元素。2015-06-06AngularJS基礎(chǔ)學(xué)習(xí)筆記之指令
指令(Directives)是所有AngularJS應(yīng)用最重要的部分。盡管AngularJS已經(jīng)提供了非常豐富的指令,但還是經(jīng)常需要?jiǎng)?chuàng)建應(yīng)用特定的指令。這篇教程會(huì)為你講述如何自定義指令,以及介紹如何在實(shí)際項(xiàng)目中使用。2015-05-05ng-events類似ionic中Events的angular全局事件
這篇文章主要介紹了ng-events類似ionic中Events的angular全局事件,本文給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2018-09-09Angular 4依賴注入學(xué)習(xí)教程之ClassProvider的使用(三)
這篇文章主要給大家介紹了關(guān)于Angular 4依賴注入之ClassProvider使用的相關(guān)資料,文中介紹的非常詳細(xì),對大家學(xué)習(xí)或者使用Angular 4具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考借鑒,下面來一起看看吧。2017-06-06angularJS 發(fā)起$http.post和$http.get請求的實(shí)現(xiàn)方法
本篇文章主要介紹了angularJS 發(fā)起$http.post和$http.get請求的實(shí)現(xiàn)方法,分別介紹了$http.post和$http.get請求的方法,有興趣的可以了解一下2017-05-05AngularJS 如何在控制臺(tái)進(jìn)行錯(cuò)誤調(diào)試
本文主要介紹AngularJS 如何在控制臺(tái)進(jìn)行錯(cuò)誤調(diào)試,還不錯(cuò),分享給大家,希望給大家做一個(gè)參考。2016-06-06