rxjs Observable filter Operator 的实现原理介绍
data:image/s3,"s3://crabby-images/e0568/e0568722bda548e5f98099540cf017103d3d407d" alt="rxjs Observable filter Operator 的实现原理介绍"
看下面这段使用 filter Operator 的代码:
原始 Observable 调用 pipe,执行自定义 Operator 的逻辑,在逻辑里生成一个 filter Operator 实例。单步调试如下:
data:image/s3,"s3://crabby-images/093e1/093e11d66210646a21b4558778e55ce5f4788586" alt=""
返回一个 filterOperatorFunction:
data:image/s3,"s3://crabby-images/f24c4/f24c49e77e61acd4754418f337d49975dc7c56b7" alt=""
filter 调用返回的是一个 filterOperatorFunction:
data:image/s3,"s3://crabby-images/453b3/453b38aee5b090622471cf380a26295aff9213d4" alt=""
至此我们只完成了 pipe 两段调用的第一段:得到 filter 返回的 filterOperatorFunction.
data:image/s3,"s3://crabby-images/6a936/6a936e7678e9c2d158e871d635ffee9393cdd02c" alt=""
然后执行 pipe 操作的第二段,如上图图例所示,将原始 Observable 传入 filterOperatorFunction:
data:image/s3,"s3://crabby-images/93695/93695af9a1a69a1b9f36af870ac000a79cc71561" alt=""
这个第二段操作,就会创建新的 FilterOperator 实例,在 lift 操作里,新建一个 Observable,然后把原始的 Observable 设置为这个新 Observable 实例的 source 字段,将 FilterOperator 实例赋给原始 Observable 的 Operator 字段。
pipe 调用执行的结果,返回一个新的 Observable 给 pipe 的调用者。最后我们 subscribe 的,就是这个 pipe 返回的 Observable.
data:image/s3,"s3://crabby-images/7de7a/7de7a93f232611a57c956d76e131af0467fa2a8d" alt=""
subscribe 内部,this 就是 pipe 返回的新 Observable,operator 指向 filterOperator,source 指向 fromEvent 返回的原始 Observable:
data:image/s3,"s3://crabby-images/21a1a/21a1ade7c46e97c970ac41bfad836c49cbd54c1d" alt=""
首先利用 TypeScript 对象结构语法,object destructing,将 this.operator 字段赋给变量 Operator,然后以原始 Observable 作为上下文,调用 filter Operator.
这样,pipe 返回的 Observable 上进行订阅,会传递到原始 Observable 的 subscribe 操作:
data:image/s3,"s3://crabby-images/56c27/56c270ccc08b0cffcf94e4a7533c88d292d50bd4" alt=""
当我们在键盘上随便敲击一个字符后,触发 MouseEvent 对应的 handler,在 handler 里调用 Observer 的 next 操作。
data:image/s3,"s3://crabby-images/270c8/270c81a0207c447e9ec21f0131b5d16ba44f4df9" alt=""
next 操作最后会触发 filtersubscriber predicate 属性即应用开发人员传入 filter Operator 的匿名函数:
data:image/s3,"s3://crabby-images/dc788/dc7885c063a6752ea748d53ffb2098e4b4845691" alt=""
filter 输入参数也是一个函数,该函数的输入即 MouseEvent:
data:image/s3,"s3://crabby-images/134dd/134dd031b208944ddf1ebf9d0b325168e986e208" alt=""
fromEvent 新建的 Observable,其实现逻辑内部,同样调用了 subscriber 的 next 方法来发射值。同时 fromEvent 返回的 Observable 同其他创建操作符比如 of 的特色之处,就在于其能够响应各种事件,比如 document 的 keyup 事件。
data:image/s3,"s3://crabby-images/53d3a/53d3a6737cd1d1d907c63061066c19515aa1db66" alt=""
这种事件注册机制,是在上图 setupSubscription
函数里实现的。
注册的具体实现,采取了浏览器原生提供的 addEventListener,调用者为 document 全局对象,传入的 eventName 为 keyup,handler 即 fromEvent 内部实现,即上图绿色高亮区域内的代码。
data:image/s3,"s3://crabby-images/23d36/23d3670c1ce0389c40dd7d93d541451dd8746a82" alt=""
data:image/s3,"s3://crabby-images/4e03a/4e03a845bf7ad66e9b41ea78ee4dbbf4a509a41d" alt=""
filter Operator 接受一个 predicate 作为输入参数,返回一个类型为 MonoTypeOperatorFunction 的函数。
这里的 T 为类型参数。MonoType 对应 T,意思是单参数。
MonoTypeOperatorFunction 是一种特殊的 OperatorFunction,当后者的输入和返回参数都相同时,即特殊化为单类型 OperatorFunction:
data:image/s3,"s3://crabby-images/5711c/5711cf0a7a8e3d97f0ddbfdbde3af249360636d9" alt=""
OperatorFunction 又是一种特殊的 UnaryFunction(一元函数),当一元函数的输入和输出类型都是一个 Observable 时,即特殊化成了 OperatorFunction.
data:image/s3,"s3://crabby-images/f0e14/f0e14eb7f3069f604dfa38f55bbd031fd524f533" alt=""
版权声明: 本文为 InfoQ 作者【Jerry Wang】的原创文章。
原文链接:【http://xie.infoq.cn/article/01e4618f6135af6325755e49d】。文章转载请联系作者。
评论