当前位置: 首页 > news >正文

使用 RxJS 库实现响应式编程

什么是 RxJS?

RxJS(Reactive Extensions for JavaScript)是一个用于响应式编程的库,它使得处理异步数据流变得更加简单和优雅。通过使用 Observables(可观察对象),你可以轻松地处理事件、HTTP 请求、定时器等异步数据源。

基本概念

在深入使用 RxJS 之前,我们需要了解几个基本概念:

  1. Observable(可观察对象):表示一个可以被观察的数据流。
  2. Observer(观察者):一个对象,它定义了如何在 Observable 发出新数据时做出反应。
  3. Subscription(订阅):当你订阅一个 Observable 时,你会得到一个 Subscription 对象,它可以用来取消订阅。
  4. Operators(操作符):用于处理 Observable 的函数,例如 map、filter、mergeMap 等。

安装 RxJS

npm install rxjs

一个简单例子

下面看一下怎么使用RxJS,首先我们可以使用 new Observable 来创建一个新的 Observable

import { Observable } from 'rxjs';const observable = new Observable(subscriber => {subscriber.next('Hello');subscriber.next('World');subscriber.complete();
});

在这个例子中,我们创建了一个 Observable,它会依次发出 “Hello” 和 “World” 字符串,然后完成。

接下来我们就可以订阅 Observable ,响应事件了,这里只是打印事件消息。

observable.subscribe({next(x) { console.log('Received: ' + x); },error(err) { console.error('Error: ' + err); },complete() { console.log('Done'); }
});

执行上面的代码,我们会得到下面的输出结果:

Received: Hello
Received: World
Done

一个安装例子

这里看一个应用场景,我们需要执行一个安装操作,这个安装可能需要执行多个步骤,比如下载文件、解压文件、安装依赖、安装资源文件等,我们可以使用 RxJS 来处理这些步骤

这里我们假定要执行下面这些步骤

  1. download(file: string)
  2. uncompress(file: string, workDir: string)
  3. installDependencies(workDir: string)
  4. installResources(workDir: string)
  5. clean(workDir: string)
import { Observable, of, from, interval } from 'rxjs';
import { map, take, concatMap, delay, tap, catchError } from 'rxjs/operators';// 执行状态
interface InstallProcessing {success: boolean;message: string;progress: number;
}class Installer {file = '/path/to/file';workDir = '/path/to/workdir';install(): Observable<InstallProcessing> {return new Observable<InstallProcessing>((observer) => {const steps = [{name: '下载安装包',method: () => this.download(this.file),},{name: '解压安装包',method:  () => this.uncompress(this.file, this.workDir),},{name: '安装依赖',method:  () => this.installDependencies(this.workDir),},{name: '安装资源文件',method:  () => this.installResources(this.workDir),},{name: '清理',method:  () => this.clean(this.workDir),},];const totalSteps = steps.length;let completedSteps = 0;const executeStep = async (index: number) => {if (index < totalSteps) {observer.next({success: false,message: `开始执行: ${steps[index].name}`,progress: (completedSteps / totalSteps) * 100,});try {await steps[index].method();completedSteps++;observer.next({success: completedSteps === totalSteps,message: `${(completedSteps / totalSteps) * 100} 完成`,progress: (completedSteps / totalSteps) * 100,});executeStep(index + 1); // 执行下一步骤} catch (error) {observer.next({success: false,message: `步骤失败: ${steps[index].name} - ${error.message}`,progress: (completedSteps / totalSteps) * 100,});observer.complete();}} else {observer.complete();}};executeStep(0); // 从第一个步骤开始});}async download(file: string): Promise<void> {// 模拟异步下载安装包console.log(`Downloading ${file}...`);await new Promise((resolve) => setTimeout(resolve, 1000)); // 模拟下载延迟}async uncompress(file: string, workDir: string): Promise<void> {// 模拟解压安装包console.log(`Uncompressing ${file} in ${workDir}...`);await new Promise((resolve) => setTimeout(resolve, 1000)); // 模拟解压延迟}async installDependencies(workDir: string): Promise<void> {// 模拟安装脚本的逻辑console.log(`Installing dependencies in ${workDir}...`);await new Promise((resolve) => setTimeout(resolve, 1000)); // 模拟安装延迟}async installResources(workDir: string): Promise<void> {// 模拟安装定时任务的逻辑console.log(`Installing resources in ${workDir}...`);await new Promise((resolve) => setTimeout(resolve, 1000)); // 模拟安装延迟}clean(workDir: string): void {// 模拟清理的逻辑console.log(`Cleaning up ${workDir}...`);}
}

使用下面的代码执行上面的安装过程并处理安装事件

const installer = new Installer();
installer.install().subscribe({next: (result) => console.log(`进度: ${result.progress}% - ${result.message}`),error: (err) => console.error(err),complete: () => console.log('Installation process completed'),
});

执行结果如下:

进度: 0% - 开始执行: 下载安装包
Downloading /path/to/file...
进度: 20% - 20 完成
进度: 20% - 开始执行: 解压安装包
Uncompressing /path/to/file in /path/to/workdir...
进度: 40% - 40 完成
进度: 40% - 开始执行: 安装依赖
Installing dependencies in /path/to/workdir...
进度: 60% - 60 完成
进度: 60% - 开始执行: 安装资源文件
Installing resources in /path/to/workdir...
进度: 80% - 80 完成
进度: 80% - 开始执行: 清理
Cleaning up /path/to/workdir...
进度: 100% - 100 完成
Installation process completed

相关文章:

使用 RxJS 库实现响应式编程

什么是 RxJS&#xff1f; RxJS&#xff08;Reactive Extensions for JavaScript&#xff09;是一个用于响应式编程的库&#xff0c;它使得处理异步数据流变得更加简单和优雅。通过使用 Observables&#xff08;可观察对象&#xff09;&#xff0c;你可以轻松地处理事件、HTTP …...

ARP攻击的原理和实现 (网络安全)

ARP攻击的原理和实现 ARP&#xff08;Address Resolution Protocol&#xff0c;地址解析协议&#xff09;是一种网络协议&#xff0c;用于在局域网内将IP地址映射到MAC地址。在以太网中&#xff0c;设备通过广播ARP请求来查询目标IP地址对应的MAC地址&#xff0c;从而建立通信…...

chatgpt model spec 2024

概述 这是模型规范的初稿&#xff0c;该文档规定了我们在OpenAI API和ChatGPT中的模型的期望行为。它包括一组核心目标&#xff0c;以及关于如何处理冲突目标或指令的指导。 我们打算将模型规范作为研究人员和数据标注者创建数据的指南&#xff0c;这是一种称为从人类反馈中进…...

单片机-LED实验

1、51工程模版 #include "reg52.h" void main(){ while(1){ } } 2、LED灯亮 #include "reg52.h" sbit LED1P2^0; void main(){ while(1){ LED10; } } 3、LED闪烁 #include "reg52.h" sbit LED1P2^0; //P2大…...

QILSTE H10-C321HRSYYA高亮红光和黄光LED灯珠

在深入探讨H10-C321HRSYYA型号的复杂特性之前&#xff0c;我们首先需要明确其基本参数和功能。这款型号的LED产品以其独特的双色设计和卓越的性能在众多同类产品中脱颖而出。其外观尺寸为3.0x1.0x2.1mm&#xff0c;采用高亮黄光和红光的双色组合&#xff0c;赋予了其在多种应用…...

Appium(一)--- 环境搭建

一、Android自动化环境搭建 1、JDK 必须1.8及以上(1) 安装&#xff1a;默认安装(2) 环境变量配置新建JAVA_HOME:安装路径新建CLASSPath%JAVA_HOME%\lib\dt.jar;%JAVA_HOME%\lib\tools.jar在path中增加&#xff1a;%JAVA_HOME%\bin;%JAVA_HOME%\jre\bin&#xff1b;(3) 验证…...

量子力学复习

黑体辐射 热辐射 绝对黑体&#xff1a; &#xff08;辐射能力很强&#xff0c;完全的吸收体&#xff0c;理想的发射体&#xff09; 辐射实验规律&#xff1a; 温度越高&#xff0c;能量越大&#xff0c;亮度越亮 温度越高&#xff0c;波长越短 光电效应 实验装置&#xf…...

22408操作系统期末速成/复习(考研0基础上手)

第一部分:计算题&#xff1a; 考察范围&#xff1a;&#xff08;标红的是重点考&#xff09; 第一章&#xff1a;CPU利用率&#xff1a; 第二章&#xff1a; 进程调度算法&#xff08;需要注意不同调度算法的优先级和题目中给出的是否可以抢占【分为可抢占和不可抢占&#xff…...

两种分类代码:独热编码与标签编码

目录 一、说明 二、理解分类数据 2.1 分类数据的类型&#xff1a;名义数据与序数数据 2.2 为什么需要编码 三、什么是独热编码&#xff1f; 3.1 工作原理&#xff1a;独热编码背后的机制 3.2 应用&#xff1a;独热编码的优势 四、什么是标签编码&#xff1f; 4.1 工作原理&…...

51单片机——共阴数码管实验

数码管中有8位数字&#xff0c;从右往左分别为LED1、LED2、...、LED8&#xff0c;如下图所示 如何实现点亮单个数字&#xff0c;用下图中的ABC来实现 P2.2管脚控制A&#xff0c;P2.3管脚控制B&#xff0c;P2.4管脚控制C //定义数码管位选管脚 sbit LSAP2^2; sbit LSBP2^3; s…...

【开源社区openEuler实践】rust_shyper

title: 探索 Rust_Shyper&#xff1a;系统编程的新前沿 date: ‘2024-12-30’ category: blog tags: Rust_ShyperRust 语言系统编程性能与安全 sig: Virt archives: ‘2024-12’ author:way_back summary: Rust_Shyper 作为基于 Rust 语言的创新项目&#xff0c;在系统编程领域…...

LiteFlow 流程引擎引入Spring boot项目集成pg数据库

文章目录 官网地址简要项目引入maven 所需jar包配置 PostgreSQL 数据库表使用LiteFlow配置 yml 文件通过 代码方式使用 liteflow数据库sql 数据在流程中周转 官网地址 https://liteflow.cc/ 简要 如果你要对复杂业务逻辑进行新写或者重构&#xff0c;用LiteFlow最合适不过。…...

阻抗(Impedance)、容抗(Capacitive Reactance)、感抗(Inductive Reactance)

阻抗&#xff08;Impedance&#xff09;、容抗&#xff08;Capacitive Reactance&#xff09;、感抗&#xff08;Inductive Reactance&#xff09; 都是交流电路中描述电流和电压之间关系的参数&#xff0c;但它们的含义、单位和作用不同。下面是它们的定义和区别&#xff1a; …...

旷视科技Java面试题及参考答案

讲一下进程间的通讯方式(如管道、消息队列、共享内存、Socket 等),各有什么特点? 管道(Pipe) 管道是最早出现的进程间通信方式之一,主要用于具有亲缘关系(父子进程)的进程之间通信。 特点: 半双工通信,数据只能单向流动。例如,在一个简单的父子进程通信场景中,父进…...

reactor的Hooks.enableAutomaticContextPropagation();不生效解决方案

1. pom中需要先增加如下的内容 <dependency><groupId>io.micrometer</groupId><artifactId>context-propagation</artifactId><version>1.1.2</version> </dependency> 2. 注意&#xff0c;要看idea是否将context-propagati…...

DS复习提纲模版

数组的插入删除 int SeqList::list_insert(int i, int item) { //插入if (i < 1 || i > size 1 || size > maxsize) {return 0; // Invalid index or list is full}for (int j size-1; j > i-1; j--) { // Shift elements to the rightlist[j1] list[j];}li…...

蓝桥杯备赛:C++基础,顺序表和vector(STL)

目录 一.C基础 1.第一个C程序&#xff1a; 2.头文件&#xff1a; 3.cin和cout初识&#xff1a; 4.命名空间&#xff1a; 二.顺序表和vector&#xff08;STL&#xff09; 1.顺序表的基本操作&#xff1a; 2.封装静态顺序表&#xff1a; 3.动态顺序表--vector&#xff1a;…...

【LLM】概念解析 - Tensorflow/Transformer/PyTorch

背景 本文将从算法原理、适用范围、强项、知名大模型的应用、python 调用几个方面&#xff0c;对深度学习框架 TensorFlow、PyTorch 和基于深度学习的模型 Transformer 进行比较。主要作用是基础概念扫盲。 一、 算法原理对比 Transformer Transformer 是一种基于深度学习的…...

对一段已知行情用python中画出K线图~

1. 已知行情: 2024/09/05 ~ 2025/1/3 date open high low close 0 2024-09-05 2785.2635 2796.0186 2777.4710 2788.3141 1 2024-09-06 2791.7645 2804.0932 2765.6394 2765.8066 2 2024-09-09 2754.7237 2756.5560 2726.9667 2736.…...

Rocky Linux下安装meld

背景介绍&#xff1a; meld是一款Linux系统下的用于 文件夹和文件的比对软件&#xff0c;非常常用&#xff1b; 故障现象&#xff1a; 输入安装命令后&#xff0c;sudo yum install meld&#xff0c;报错。 12-31 22:12:17 ~]$ sudo yum install meld Last metadata expirat…...

多模态2025:技术路线“神仙打架”,视频生成冲上云霄

文&#xff5c;魏琳华 编&#xff5c;王一粟 一场大会&#xff0c;聚集了中国多模态大模型的“半壁江山”。 智源大会2025为期两天的论坛中&#xff0c;汇集了学界、创业公司和大厂等三方的热门选手&#xff0c;关于多模态的集中讨论达到了前所未有的热度。其中&#xff0c;…...

LeetCode - 394. 字符串解码

题目 394. 字符串解码 - 力扣&#xff08;LeetCode&#xff09; 思路 使用两个栈&#xff1a;一个存储重复次数&#xff0c;一个存储字符串 遍历输入字符串&#xff1a; 数字处理&#xff1a;遇到数字时&#xff0c;累积计算重复次数左括号处理&#xff1a;保存当前状态&a…...

在 Nginx Stream 层“改写”MQTT ngx_stream_mqtt_filter_module

1、为什么要修改 CONNECT 报文&#xff1f; 多租户隔离&#xff1a;自动为接入设备追加租户前缀&#xff0c;后端按 ClientID 拆分队列。零代码鉴权&#xff1a;将入站用户名替换为 OAuth Access-Token&#xff0c;后端 Broker 统一校验。灰度发布&#xff1a;根据 IP/地理位写…...

【快手拥抱开源】通过快手团队开源的 KwaiCoder-AutoThink-preview 解锁大语言模型的潜力

引言&#xff1a; 在人工智能快速发展的浪潮中&#xff0c;快手Kwaipilot团队推出的 KwaiCoder-AutoThink-preview 具有里程碑意义——这是首个公开的AutoThink大语言模型&#xff08;LLM&#xff09;。该模型代表着该领域的重大突破&#xff0c;通过独特方式融合思考与非思考…...

第25节 Node.js 断言测试

Node.js的assert模块主要用于编写程序的单元测试时使用&#xff0c;通过断言可以提早发现和排查出错误。 稳定性: 5 - 锁定 这个模块可用于应用的单元测试&#xff0c;通过 require(assert) 可以使用这个模块。 assert.fail(actual, expected, message, operator) 使用参数…...

VTK如何让部分单位不可见

最近遇到一个需求&#xff0c;需要让一个vtkDataSet中的部分单元不可见&#xff0c;查阅了一些资料大概有以下几种方式 1.通过颜色映射表来进行&#xff0c;是最正规的做法 vtkNew<vtkLookupTable> lut; //值为0不显示&#xff0c;主要是最后一个参数&#xff0c;透明度…...

Java入门学习详细版(一)

大家好&#xff0c;Java 学习是一个系统学习的过程&#xff0c;核心原则就是“理论 实践 坚持”&#xff0c;并且需循序渐进&#xff0c;不可过于着急&#xff0c;本篇文章推出的这份详细入门学习资料将带大家从零基础开始&#xff0c;逐步掌握 Java 的核心概念和编程技能。 …...

PAN/FPN

import torch import torch.nn as nn import torch.nn.functional as F import mathclass LowResQueryHighResKVAttention(nn.Module):"""方案 1: 低分辨率特征 (Query) 查询高分辨率特征 (Key, Value).输出分辨率与低分辨率输入相同。"""def __…...

C#学习第29天:表达式树(Expression Trees)

目录 什么是表达式树&#xff1f; 核心概念 1.表达式树的构建 2. 表达式树与Lambda表达式 3.解析和访问表达式树 4.动态条件查询 表达式树的优势 1.动态构建查询 2.LINQ 提供程序支持&#xff1a; 3.性能优化 4.元数据处理 5.代码转换和重写 适用场景 代码复杂性…...

Go语言多线程问题

打印零与奇偶数&#xff08;leetcode 1116&#xff09; 方法1&#xff1a;使用互斥锁和条件变量 package mainimport ("fmt""sync" )type ZeroEvenOdd struct {n intzeroMutex sync.MutexevenMutex sync.MutexoddMutex sync.Mutexcurrent int…...