#1 可观察者对象,观察者,操作符初步了解

可观察对象和观察者模式的差别:

  • 可观察者对象不启动流媒体项目(streaming items),直到至少有一个观察者订阅(Observer)它(这句话很重要)
  • 和迭代器类似,一个可观察者对象当序列(sequence)完成时能发送信号(signal)
  • 使用可观察者,我们能够声明如何响应它发出的序列元素,而不是响应当单独的个体元素
  • 我们可以高效的复制,转换和查询序列,这些操作对所有序列元素都有效

Observables && Observer

有很多种方式创建Observables对象,使用 create 方法(在Rxjs中称之为操作符)是最明显的方式。

Rx.Observable接收一个以Observer作为参数的回调,这个回调函数定义可观察对象如何发出值(emit values).

var observable = Rx.Observable.create(function(observer) {
  observer.onNext('James');
  observer.onNext('Jordan');
  observer.onNext('Durant');
  observer.onCompleted(); // 完成
})

当订阅这个可观察对象,它将调用 onNext 方法, 生成3个字符串,然后调用 onCompleted 发出信号,表示序列完成。我们使用 Observers 来订阅 Observable

Observers

Observers订阅可观察对象,当可观察对象发生变化,观察者调用相关的方法,它有3个方法:

  • onNext: 相当于观察者模式中的 ``Update 方法
  • onError: 可观察对象发生错误时调用,它调用之后,再调用onNext是无效的
  • onCompleted: 发出信号,没有更多数据了

创建观察者:

var observer = Rx.Observer.create(
  function onNext(x) { console.log('next: ' + x);},
  function onError(err) { console.log('Error: ' + err); },
  function onCompleted() { console.log('Done'); }
);

Rx.Observer.create接收函数 onNextonErroronCompleted, 返回一个Observer实例,这些方法是可选的。比如有些无限序列(比如用户一直点击鼠标),onCompleted是永远不会被调用的,如果有信心不发生错误,onError也是可以不用写的。

Observable ajax请求

使用可观察对象来封装一个ajax请求,返回一个可观察对象实例:

function getUrl(url) {
  return Rx.Observable.create(function(observer) {
    var xhr = new XMLHttpRequest();
    xhr.open('GET', url);

    xhr.onload = function() {
      if (req.status === 200) {
        observer.onNext(xhr.response);
        observer.onCompleted();
      } else {
        observer.onError(new Error(xhr.statusText));
      }
    };

    xhr.onerror = function() {
      observer.onError(new Error('未知错误'));
    };

    xhr.send();
  })
}

光有可观察对象,没有观察者,流是不会触发的,下面是观察者:

var test = getUrl('https://randomuser.me/api/')

// 订阅一个观察者
test.subscribe(
  function onNext(x) {console.log('Result: ' + x)},
  function onError(err) {console.log('Error: ' + err)},
  function onCompleted() {console.log('done')}
);

此处我们可以发现,我们没有通过Rx.Observer.create显式的创建一个Observer, 而是通过 subscribe操作符来创建Observer

操作符

在Rxjs中,转换或查询序列方法被称之为 operators(操作符)。上面的ajax封装可以使用已经存在的rxjs-dom库来完成。

// 首先先引入rxjs-dom.js库
// https://cdnjs.cloudflare.com/ajax/libs/rxjs-dom/7.0.3/rx.dom.compat.min.js

Rx.DOM.get('https://randomuser.me/api/')
  .subscribe(
    function onNext(data) {console.log(data.response)},
    function onError(err) {console.log('Error: ' + err)},   
  )

在Rxjs编程中,可以把所有的数据都当作可观察者对象,不仅是异步数据,比如数组等

常见的操作符

1.将数组变为可观察者对象: from(arr)

Rx.Observable
  .from(['a', 'b', 'c'])
  .subscribe(
    function(x) {console.log('Next: ' + x);}, // 可以不用写方法名'onNext'
    function(err) {console.log('Error: ' + err);},
    function() {console.log('完成');}
  )

//结果
Next: a
Next: b
Next: c
完成

2.将事件变为可观察者对象: fromEvent(dom, eventType)

var allMoves = Rx.Observable
                .fromEvent(document, 'mousemove')

// 订阅
allMoves.subscribe(
  function(e) {console.log(e.clientX, e.clientY);}
)

// 另外还可以对可观察者对象进行转换过滤等
// 在allMoves基础上
var movesOnTheLeft = allMoves.filter(function(e) 
      {return e.clientX < window.innerWidth/2}
    )
movesOnTheLeft.subscribe(function(e) {console.log('鼠标在左边',e.clientX)})

3.将回调函数变为可观察者对象: fromCallback() | fromNodeCallback()

比如Nodejs中

var Rx = require('rx');
var fs = require('fs');

// 从readdir方法中创建可观察者对象
var readdir = Rx.Observable.fromNodeCallback(fs.readdir);

var source = readdir('/users/someone');

var subscription = source.subscribe(
    function(res) {console.log('目录: ' + res);}, // 可以不用写方法名'onNext'
    function(err) {console.log('Error: ' + err);},
    function() {console.log('完成');}
  )

总结

初步认识Rxjs,了解什么是可观察对象,观察者,如何使用操作符来创建它们,本文主要使用 RxJS4.10版本,注意和5.0的区别。

使用到了 rxjs-dom 库来完成ajax请求,初步了解的一些操作符:

  • create
  • subscribe
  • from
  • fromEvent
  • fromNodeCallback

可观察者的2个方法(接受以observer为参数的回调函数):

  • observer.onNext()
  • observer.onCompleted()

观察者的3个函数:

  • function onNext() {}
  • function onError() {}
  • function onCompleted() {}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,547评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,399评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,428评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,599评论 1 274
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,612评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,577评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,941评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,603评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,852评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,605评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,693评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,375评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,955评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,936评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,172评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,970评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,414评论 2 342

推荐阅读更多精彩内容