简单模拟实现Rxjs Observable
1.先定义类型
export type Observer = {
next: (any) => void,
complete?: (any) => void,
}
export interface OnSubscribeAction {
(observer: Observer): void;
}
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的更多相关文章
- java web学习总结(二十二) -------------------简单模拟SpringMVC
在Spring MVC中,将一个普通的java类标注上Controller注解之后,再将类中的方法使用RequestMapping注解标注,那么这个普通的java类就够处理Web请求,示例代码如下: ...
- WPF简单模拟QQ登录背景动画
介绍 之所以说是简单模拟,是因为我不知道QQ登录背景动画是怎么实现的.这里是通过一些办法把它简化了,做成了类似的效果 效果图 大体思路 首先把背景看成是一个4行8列的点的阵距,X轴Y轴都是距离70.把 ...
- Linux 内核 链表 的简单模拟(2)
接上一篇Linux 内核 链表 的简单模拟(1) 第五章:Linux内核链表的遍历 /** * list_for_each - iterate over a list * @pos: the & ...
- Linux 内核 链表 的简单模拟(1)
第零章:扯扯淡 出一个有意思的题目:用一个宏定义FIND求一个结构体struct里某个变量相对struc的编移量,如 struct student { int a; //FIND(struct stu ...
- JavaWeb学习总结(四十九)——简单模拟Sping MVC
在Spring MVC中,将一个普通的java类标注上Controller注解之后,再将类中的方法使用RequestMapping注解标注,那么这个普通的java类就够处理Web请求,示例代码如下: ...
- 简单模拟Hibernate的主要功能实现
在学习期间接触到Hibernate框架,这是一款非常优秀的O/R映射框架,大大简化了在开发web项目过程中对数据库的操作.这里就简单模拟其底层的实现. /*******代码部分,及其主要注解***** ...
- 【HDU 4452 Running Rabbits】简单模拟
两只兔子Tom和Jerry在一个n*n的格子区域跑,分别起始于(1,1)和(n,n),有各自的速度speed(格/小时).初始方向dir(E.N.W.S)和左转周期turn(小时/次). 各自每小时往 ...
- Jquery源码分析与简单模拟实现
前言 最近学习了一下jQuery源码,顺便总结一下,版本:v2.0.3 主要是通过简单模拟实现jQuery的封装/调用.选择器.类级别扩展等.加深对js/Jquery的理解. 正文 先来说问题: 1. ...
- (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 ...
随机推荐
- 搭建环境-git常见使用总结
Descripton:git 一.Git安装和本地用户全局配置 官网下载并且安装 查看是否安装成功win + R输入git,出现git命令指南,则安装成功 全局配置本地用户,在git Bash中进行下 ...
- 数据结构和算法(Golang实现)(14)常见数据结构-栈和队列
栈和队列 一.栈 Stack 和队列 Queue 我们日常生活中,都需要将物品排列,或者安排事情的先后顺序.更通俗地讲,我们买东西时,人太多的情况下,我们要排队,排队也有先后顺序,有些人早了点来,排完 ...
- python成功安装torch模块
最近项目要使用到torch模块,但是在安装的过程中发现torch直接使用pip install安装是安装不成功的.然后就百度,发现并没有什么卵用,所以就google一番,不禁感叹,这种新的东西,还是外 ...
- G - Harmonic Number (II) LightOJ - 1245
算是一个找规律的题目吧. 枚举前sqrt(n)个数,数i出现的次数为n/i-n/(i+1),对答案的贡献为(n/i-n/(i+1))*i. 对于sqrt后边的数,可以直接由n/i获得,并且一定只出现一 ...
- G - Can you find it? 二分
Give you three sequences of numbers A, B, C, then we give you a number X. Now you need to calculate ...
- 今天我们来谈谈绝对定位和相对定位的区别,和需要注意的问题;position:absolute|relative;
首先position:absolute|relative; 前者是绝对定位,后者是相对定位: position属性的四个值: static,relative,fixed,absolute; 重点重点重 ...
- 视频+图文 String类干货向总结
目录 一.字符串的介绍及视频讲解 二.字符串的两种创建方式 方法一:通过new创建 方法二:直接创建 三.字符串的长度获取:length()方法 四.使用 == 和equals()方法比较两个字符串 ...
- 从一次“并发修改字段业务”引出多版本并发控制与InnoDB锁
并发字段修改业务 最近在主要在做"工作流引擎"课题的预研工作,在涉及到"会签任务"(工作流业务概念,这与我们今天讨论文问题没有太多关联)的时候,遇到了一个并发修 ...
- SpringCloud(四)学习笔记之Feign
Feign是一个声明式的Web服务客户端,可帮助我们更加便捷.优雅地调用HTTP API Feign可以与Eureka和Ribbon组合使用以支持负载均衡 一.构建Eureka Server [基于第 ...
- cli命令速查
在文件的指定行(n)插入指定内容: sed -i "niecho "haha"" a 执行后,在a文件的第n行插入echo "haha" 多 ...