1.先定义类型

export type Observer = {
next: (any) => void,
complete?: (any) => void,
} export interface OnSubscribeAction {
(observer: Observer): void;
}

 export interface mapFun {
      (x: any): any
  }
 

2.自定义类

export class MyObservable {
private subscribeAction: OnSubscribeAction;
constructor(subscribeAction: OnSubscribeAction) {
this.subscribeAction = subscribeAction;
} subscribe(observer: Observer) {
return this.subscribeAction(observer)
}
}

3.实例

const onSubscribe: OnSubscribeAction = (observer: Observer) => {
observer.next(1);
observer.next(2);
observer.next(3);
} const theObserver = {
next: console.log,
} const source$ = new MyObservable(onSubscribe);
source$.subscribe(theObserver);

结果一次输出1,2,3

4.增加map操作符

export class MyObservable {
private subscribeAction: OnSubscribeAction;
constructor(subscribeAction: OnSubscribeAction) {
this.subscribeAction = subscribeAction;
} subscribe(observer: Observer) {
return this.subscribeAction(observer)
}
  //rxjs 的每个操作符都会返回一个新的Observable
map(fun: mapFun) {
const nextObserver = new MyObservable((observer: Observer) => {
this.subscribe({
next(item){
observer.next(fun(item))
}
})
})
return nextObserver
}
}

5.再次调用

const onSubscribe: OnSubscribeAction = (observer: Observer) => {
observer.next(1);
observer.next(2);
observer.next(3);
} const theObserver = {
next: console.log,
} const source$ = MyObservable.create(onSubscribe);
source$.map(x => x * 2).map(x => x + 1).subscribe(theObserver);

结果一次输出3,5,7

=====================================

更进一步把map作为一个单独的函数抽离

export function map(fun: mapFun) {
const nextObserver = new MyObserver((observer: Observer) => {
this.subscribe({
next(item){
observer.next(fun(item))
}
})
})
return nextObserver
}

完整代码如下所示

export type Observer = {
next: (any) => void,
complete?: (any) => void,
} export interface OnSubscribeAction {
(observer: Observer): void;
} export interface mapFun {
(x: any): any
} export function map(fun: mapFun) {
const nextObserver = new MyObservable((observer: Observer) => {
this.subscribe({
next(item){
observer.next(fun(item))
}
})
})
return nextObserver
} export class MyObservable {
private subscribeAction: OnSubscribeAction;
constructor(subscribeAction: OnSubscribeAction) {
this.subscribeAction = subscribeAction;
} static create(subscribeAction: OnSubscribeAction) {
return new MyObservable(subscribeAction)
} subscribe(observer: Observer) {
return this.subscribeAction(observer)
}
}

由于map函数中使用到this,需要用call或者apply或者bind改变其指向,使用示例如下

 const onSubscribe: OnSubscribeAction = (observer: Observer) => {
observer.next(1);
observer.next(2);
observer.next(3);
} const theObserver = {
next: console.log,
} const source$ = new MyObservable(onSubscribe);
const subScribtion$ = map.call(source$, x => x * 2);
map.call(subScribtion$, x => x + 1).subscribe(theObserver);

输出结果依次为3,5,7

=====================

由于在map函数中需要访问this, 导致map不是一个纯函数。在map中的this指代上一个observable,现在改为通过参数传递的形式把上一个observable传递给下一个observable。

map函数修改如下

export function map(fun: mapFun) {
return function(obs$) {
const nextObserver = new MyObservable((observer: Observer) => {
obs$.subscribe({
next(item){
observer.next(fun(item))
}
})
})
return nextObserver
}
}

同时MyObservable自定义类需要增加一个pipe方法

export class MyObservable {
private subscribeAction: OnSubscribeAction;
constructor(subscribeAction: OnSubscribeAction) {
this.subscribeAction = subscribeAction;
} subscribe(observer: Observer) {
return this.subscribeAction(observer)
} pipe(fn: any){
return fn(this);
}
}

最终的调用方式如下

const onSubscribe: OnSubscribeAction = (observer: Observer) => {
observer.next(1);
observer.next(2);
observer.next(3);
} const theObserver = {
next: console.log,
} const source$ = new MyObservable(onSubscribe);
source$.pipe(map(x => x*2)).pipe(map(x => x + 1)).subscribe(theObserver)

结果依次输出3,5,7

简单模拟实现Rxjs Observable的更多相关文章

  1. java web学习总结(二十二) -------------------简单模拟SpringMVC

    在Spring MVC中,将一个普通的java类标注上Controller注解之后,再将类中的方法使用RequestMapping注解标注,那么这个普通的java类就够处理Web请求,示例代码如下: ...

  2. WPF简单模拟QQ登录背景动画

    介绍 之所以说是简单模拟,是因为我不知道QQ登录背景动画是怎么实现的.这里是通过一些办法把它简化了,做成了类似的效果 效果图 大体思路 首先把背景看成是一个4行8列的点的阵距,X轴Y轴都是距离70.把 ...

  3. Linux 内核 链表 的简单模拟(2)

    接上一篇Linux 内核 链表 的简单模拟(1) 第五章:Linux内核链表的遍历 /** * list_for_each - iterate over a list * @pos: the & ...

  4. Linux 内核 链表 的简单模拟(1)

    第零章:扯扯淡 出一个有意思的题目:用一个宏定义FIND求一个结构体struct里某个变量相对struc的编移量,如 struct student { int a; //FIND(struct stu ...

  5. JavaWeb学习总结(四十九)——简单模拟Sping MVC

    在Spring MVC中,将一个普通的java类标注上Controller注解之后,再将类中的方法使用RequestMapping注解标注,那么这个普通的java类就够处理Web请求,示例代码如下: ...

  6. 简单模拟Hibernate的主要功能实现

    在学习期间接触到Hibernate框架,这是一款非常优秀的O/R映射框架,大大简化了在开发web项目过程中对数据库的操作.这里就简单模拟其底层的实现. /*******代码部分,及其主要注解***** ...

  7. 【HDU 4452 Running Rabbits】简单模拟

    两只兔子Tom和Jerry在一个n*n的格子区域跑,分别起始于(1,1)和(n,n),有各自的速度speed(格/小时).初始方向dir(E.N.W.S)和左转周期turn(小时/次). 各自每小时往 ...

  8. Jquery源码分析与简单模拟实现

    前言 最近学习了一下jQuery源码,顺便总结一下,版本:v2.0.3 主要是通过简单模拟实现jQuery的封装/调用.选择器.类级别扩展等.加深对js/Jquery的理解. 正文 先来说问题: 1. ...

  9. (hdu step 8.1.6)士兵队列训练问题(数据结构,简单模拟——第一次每2个去掉1个,第二次每3个去掉1个.知道队伍中的人数<=3,输出剩下的人 )

    题目: 士兵队列训练问题 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) Total ...

随机推荐

  1. std::chrono计算程序运行时间

    void CalRunTime() { auto t1=std::chrono::steady_clock::now(); //run code auto t2=std::chrono::steady ...

  2. 数据结构和算法(Golang实现)(23)排序算法-归并排序

    归并排序 归并排序是一种分治策略的排序算法.它是一种比较特殊的排序算法,通过递归地先使每个子序列有序,再将两个有序的序列进行合并成一个有序的序列. 归并排序首先由著名的现代计算机之父John_von_ ...

  3. Pytest系列(21)- allure的特性,@allure.description()、@allure.title()的详细使用

    如果你还想从头学起Pytest,可以看看这个系列的文章哦! https://www.cnblogs.com/poloyy/category/1690628.html 前言 前面介绍了两种allure的 ...

  4. ES6构造函数class 和 ES5构造函数语法

    构造函数就是JavaScript程序定义好的函数,我们直接使用就可以,实际也是一种函数,构造函数专门用于生成定义对象,通过构造函数生成的对象,称为实例化对象 构造函数分为两种,一种是JavaScrip ...

  5. JAVA—HashMap

    一些关于hashmap的学习笔记 1.HashMap底层实现原理 在JDK1.7中HashMap是以数组加链表的形式组成的,在JDK1.8之后新增了红黑树的组成结构,当链表大于8并且容量大于64时,链 ...

  6. k3s-初体验

    k3s安装步骤 1.准备工作 关闭swap交换分区 swapoff -a 关闭防火墙 systemctl stop firewalld.service 2.下载启动k3s包 https://githu ...

  7. L12 Transformer

    Transformer 在之前的章节中,我们已经介绍了主流的神经网络架构如卷积神经网络(CNNs)和循环神经网络(RNNs).让我们进行一些回顾: CNNs 易于并行化,却不适合捕捉变长序列内的依赖关 ...

  8. L4文本预处理

    文本预处理 timemachine.txt数据下载地址 链接:https://pan.baidu.com/s/1RO2OLyTRQZ90HJUW7V7BCQ 提取码:bjox NLTK数据集下载 链接 ...

  9. ASE课程总结 by 冯晓云

    开始的开始,采访往届ASE班的blog:http://www.cnblogs.com/legs/p/4894362.html 和北航软工M1检查:http://www.cnblogs.com/legs ...

  10. 申请elasticsearch中x-pack插件许可证及授权

    前提:         ES主机中elasticsearch x-pack插件许可证申请使用期限为1年,到期后x-pack插件将不再可用,重启elasticsearch服务后日志会提示一下警告,如图所 ...