React 项目运用 RxJS 设置节流

RxJS 核心概念

一小时入门RxJS,中文教学放心适用,讲解非常棒

按钮组:

html 复制代码
<button nz-button nzType="primary" style="cursor:pointer !important; margin-bottom: 4px;" (click)="throttleClick(flag1, 2000, btnFunc1)">btn1</button>

<button nz-button nzType="primary" style="cursor:pointer !important; margin-bottom: 4px;" (click)="throttleClick(flag2, 2000, btnFunc2)">btn2</button>

引用 rxjs :

javascript 复制代码
import { Subject, forkJoin, Observable, of } from 'rxjs'; // 引入 forkJoin
import { throttleTime, takeUntil, catchError, tap, map } from 'rxjs/operators';

按钮触发事件:

javascript 复制代码
// 点击 btn1 按钮触发的事件
btnFunc1(): Observable<any> {}

// 点击 btn2 按钮触发的事件
btnFunc1(): Observable<any> {}

设置节流:

  • takeUntil

    函数签名: takeUntil(notifier: Observable): Observable

    发出值,直到提供的 observable 发出值,它便完成。

  • throttleTime

    函数签名: throttleTime(duration: number, scheduler: Scheduler): Observable

    当指定的持续时间经过后发出最新值。

  • call() 是函数的原生方法,作用是立即执行该函数 ,并强制绑定函数内部的 this 指向第一个参数,后续参数作为函数的入参传入。

  • bind() 是 JavaScript 函数的原生方法,核心作用是创建一个新函数,这个新函数被调用时,其内部的 this 会永久绑定到 bind() 的第一个参数,同时可以预先传入部分参数(柯里化)

javascript 复制代码
throttleClick(btnKey: string, delay: number, callback: () => void): void {
   // 若该按钮未创建 Subject,初始化并设置节流
   if (!this.buttonSubjects.has(btnKey)) {
   const subject$ = new Subject<void>();
   subject$
       .pipe(
       throttleTime(delay), // 节流核心
       takeUntil(this.destroy$) // 组件销毁时取消订阅
       )
       .subscribe(function() { callback.call(this)}.bind(this)); // 执行按钮回调
       this.buttonSubjects.set(btnKey, subject$);
   }

   // 发送点击信号
   this.buttonSubjects.get(btnKey) && this.buttonSubjects.get(btnKey).next();
}

完整 js 代码

javascript 复制代码
import { Subject, forkJoin, Observable, of } from 'rxjs'; // 引入 forkJoin
import { throttleTime, takeUntil, catchError, tap, map } from 'rxjs/operators';

// 存储每个按钮的 Subject(key=按钮标识,value=对应 Subject)
private buttonSubjects = new Map<string, Subject<void>>();
// 组件销毁信号
private destroy$ = new Subject<void>();

throttleClick(btnKey: string, delay: number, callback: () => void): void {
   // 若该按钮未创建 Subject,初始化并设置节流
   if (!this.buttonSubjects.has(btnKey)) {
   const subject$ = new Subject<void>();
   subject$
       .pipe(
       throttleTime(delay), // 节流核心
       takeUntil(this.destroy$) // 组件销毁时取消订阅
       )
       .subscribe(function() { callback.call(this)}.bind(this)); // 执行按钮回调
       this.buttonSubjects.set(btnKey, subject$);
   }

   // 发送点击信号
   this.buttonSubjects.get(btnKey) && this.buttonSubjects.get(btnKey).next();
}

// 点击 btn1 按钮触发的事件
btnFunc1(): Observable<any> {}

// 点击 btn2 按钮触发的事件
btnFunc1(): Observable<any> {}


ngOnDestroy() {
    this.destroy$.next();
    this.destroy$.complete();
    // 遍历销毁所有按钮的 Subject
    this.buttonSubjects.forEach(subject$ => subject$.complete());
    this.buttonSubjects.clear();
}
相关推荐
老王以为22 分钟前
为什么 React 和 Vue 不一样?
前端·vue.js·react.js
迪菲赫尔曼3 小时前
从 0 到 1 打造工业级推理控制台:UltraConsole(Ultralytics + FastAPI + React)开源啦!
前端·yolo·react.js·计算机视觉·开源·fastapi
Highcharts.js4 小时前
React 开发实战:如何使用 useEffect 为 Highcharts 注入实时数据
前端·javascript·react.js·开发实战·实时数据·highcharts·轮询数据
光影少年5 小时前
前端SSR和ssg区别
前端·vue.js·人工智能·学习·react.js
祖国的好青年6 小时前
VS Code 搭建 React Native 开发环境(Windows 实战指南)
android·windows·react native·react.js
天天进步20157 小时前
魔音漫创源码解析:架构总览:Electron 30 + React 18 + Zustand,构建桌面级影视生产工具
react.js·架构·electron
kyriewen119 小时前
Next.js部署:从本地跑得欢,到线上飞得稳
开发语言·前端·javascript·科技·react.js·前端框架·ecmascript
朝阳3910 小时前
react【实战】首页 -- 白天/黑夜主题切换(含组件封装)
前端·react.js·前端框架
卷Java10 小时前
ReAct范式实战:让Agent学会边想边做
javascript·react.js·大模型·llm·ecmascript·multi-agent
M ? A10 小时前
Vue 转 React | VuReact编译工具快速入门
前端·javascript·vue.js·后端·react.js·面试·vureact