React 中使用 RxJS 优化数据流的处理方案

目录
  • 正文
  • 一般来说,处理组件中的数据流无非三种情况:
    • 下面我们看一个很简单的例子:
    • 那么,问题来了,使用数据流的方式来处理数据有什么好处呢?

正文

现在我们比较熟悉的是使用 functional component 和 hooks 来处理 react 逻辑。熟悉 Angular 的用户可能比较熟悉通过 RxJS 来处理异步和数据状态。那么我们看一下 React 中使用 RxJS 会不会有什么优势呢?

一般来说,处理组件中的数据流无非三种情况:

  • Component 中 props 的变化
  • 用户在界面上操作,触发 event 修改 component 的数据
  • Component 中监听的 store 触发数据变化

下面我们看一个很简单的例子:

有一个 component,显示 input 中输入的数据。逻辑也很简单:

export function ShowInput(props: { data: string; }): JSX.Element {
  return <>{props.data}</>;
}

也就是第一种情况,component 的状态是通过,props 传递进来的。

这当然没有问题,当数据变化的时候重新渲染即可。假设我们每一次输入就会触发一次 api call,然后把 api 返回的内容显示在界面上。这里就需要处理两个问题:

  • 尽可能减少 api call 的次数(比如用户输入有 500ms 的停顿以上时,我们认为用户已经输入结束,才开始 call api)
  • 用户如果多次输入,会有多次 api call,那么后面 api 返回的数据应该永远覆盖前面 api 返回的数据。
  • 暂时不考虑 api 可能出错,需要重新 call api 的情况

熟悉 RxJS 的话,会知道,这是一个非常简单的数据流的问题,如果我们把 props 的数据看作是一个数据流的话,就可以非常简单的解决了。

const apiRespons$ = data$.pipe(
  debounceTime(500),
  switchMap(data => getApiResponse(data)),
);
// Mock API call
function getApiResponse(data: string) {
  return of(data + ' data from api').pipe(delay(1000));
}

那么,问题就来了,如何将 props data 转换成 stream, 又如何将处理完的 stream 转换成 component state 呢。

这里就需要引入一个库: rxjs-hooks

用法一: 将 stream 转换为 componet state.

比如我们已经得到了 apiResponse$,如何在 component 中显示的呢?

const apiRespons$ = data$.pipe(
  debounceTime(500),
  switchMap(data => getApiResponse(data)),
);
const apiResponse = useObservable(() => apiRespons$);

那么如何结合 props 的变化使用呢?

用法二: 将 props 转换为 stream useObservable 还可以输入两个参数:

  • 状态的初始值 (类似,useState)
  • 监听变量(类似 useEffect 的数组参数),转换为 stream 作为函数的参数

完整的代码会变成这样:

export function ShowInput(props: { data: string; }): JSX.Element {
  const apiResponse = useObservable((_, input$) => input$.pipe(
    debounceTime(500),
    switchMap(([data]) => getApiResponse(data)),
  ), '', [props.data]);
  return <>{apiResponse}</>;
}

当然,也有可能,数据的变化时当前 component 的,不是 props 传进来的。也就是说这个 input 可能会是在当前 component 中。

当然,我们简单的把 input 放在当前的 componnet 中,把监听内容从 props 换成 state. 也比较简单。

const [data, setData] = useState<string>();
const apiResponse = useObservable((_, input$) => input$.pipe(
  debounceTime(500),
  switchMap(([inputData]) => getApiResponse(inputData || '')),
), '', [data]);

那么,有没有更简单一点的方法呢?

就涉及到用法三:将 event 函数的调用自动转换为 stream

const [onInputChange, apiResponse] = useEventCallback((data$: Observable<string>) => data$.pipe(
  debounceTime(500),
  switchMap((data) => getApiResponse(data || '')),
), '');

这就将 useState 和 useObservable 合为一体。第一个参数时 event 函数,第二个参数时 state。

当然,如果有使用 redux observable 的话,就可以很好的跟 rxjs-hooks 合为一体。

那么,问题来了,使用数据流的方式来处理数据有什么好处呢?

  • RxJS 内置了很多数据流的处理方式,可以大大的简化我们处理数据的流程。类似于异步中的 lodash.
  • 在 React 中我们往往会同时使用 Redux 和 Hooks, 某种程度上说,我们会使用 Redux 处理全局或者说大的状态管理,hooks 处理 component 层,或者说小的数据状态管理。使用 redux observable 和 rxjs hooks 不失为一种沟通全局状态和局部状态的好的方式。
  • 我们直到 redux 时跨平台的。但是有些时候我们并不希望把所有的数据状态都封装在 redux 中。如果要实现数据状态的跨平台,而不使用 redux 的话,rxjs 实现一个状态关机工具无非是最简单的。因为一个 subject 就是一个最简单的 store. 让我们的主要逻辑不依赖于框架,是最好的多平台共享的方式。

对于 component 层的局部状态,逻辑共享:

react   angular
|       /
hooks  component store
|      /
js rxjs store

对于 redux 状态共享

react     angular
|         /
redux observable

以上就是React 中使用 RxJS 优化数据流的处理方案的详细内容,更多关于React RxJS 优化数据流的资料请关注我们其它相关文章!

(0)

相关推荐

  • RxJS中的Observable和Observer示例详解

    目录 引言 概念 牛刀小试 Observable Observable 剖析 Observer 结束语 引言 最近在项目当中别的小伙伴使用到了Rxjs,我一眼看上去有点懵,感觉挺复杂,挺绕的.于是抓紧补补课,然后就可以和小伙伴们一起交流怎么能优雅的使用Rxjs.由于内容比较多,会分为三篇来讲解说明 初识 RxJS中的 Observable 和 Observer 细说 RxJS中的 Operators 在谈 RxJS中的 Subject和Schedulers 概念 RxJS是一个库,可以使用可观察

  • ​​​​​​​Rxjs map, mergeMap 和 switchMap 的区别与联系

    目录 前言 map mergeMap switchMap 前言 map.mergeMap 和 switchMap 是 RxJS 中的三个主要运算符,在 SAP Spartacus 开发中有着广泛的使用场景. map map 是 Observables 中最常见的运算符. 它的作用与数组中的映射相对相似. map 接收从 Observable 发出的每个值,对其执行操作并返回一个 Observable(因此 Observable 链可以继续). 把它想象成一个函数,它将采用原始值和投影. 该函数将

  • Rxjs监听精确使用版本上线

    目录 导语 思路 注释掉一个触发条件 ...解构是2次,是不是这里导致的? 那就是监听的地方,触发了2遍 拓展: You provided an invalid object 解决办法 总结 导语 最近在开发新feature,周五需要版本合并,上线.可是从DevTool Network请求记录来看,每次界面加载完毕,异步条件触发,filter api 都是调用2遍. 根据日志,定位带有嫌疑code位置. code多请求一次,对服务器.用户体验都是不能接受的,版本是不可能上线的. 思路 问题可不可

  • JavaScript Rxjs mergeMap 的使用场合

    注意: flatMap 是 mergeMap 的别名. 如果一次只能激活一个内部订阅,请使用 switchMap. 如果内部 observables 的发射和订阅顺序很重要,请使用 concatMap. 当需要展平内部 observable 但想要手动控制内部订阅的数量时,是 mergeMap 极佳的使用场合. 例如,当使用 switchMap 时,每个内部订阅在源发出时完成,即任意时间段只允许一个活动的内部订阅.相比之下,mergeMap 允许同时激活多个内部订阅.因此,mergeMap 最常

  • 在 React 中使用 Redux 解决的问题小结

    目录 在 React 中使用 Redux 解决的问题 在 React 项目中加入 Redux 的好处 React + Redux 安装 Redux React 中 Redux 的工作流程 React 计数器案例 使用 Redux Provide 组件 connect 方法 使用 bindActionCreators 方法继续简化 代码重构 为 Action 传递参数 Redux 弹出框 初始化静态内容 添加默认隐藏状态 定义操作按钮 衍生的问题 拆分合并 reducer 拆分 合并 调整组件 在

  • 深入研究React中setState源码

    React作为一门前端框架,虽然只是focus在MVVM中的View部分,但还是实现了View和model的绑定.修改数据的同时,可以实现View的刷新.这大大简化了我们的逻辑,只用关心数据流的变化,同时减少了代码量,使得后期维护也更加方便.这个特性则要归功于setState()方法.React中利用队列机制来管理state,避免了很多重复的View刷新.下面我们来从源码角度探寻下setState机制. 1 还是先声明一个组件,从最开始一步步来寻源: class App extends Comp

  • 关于react中组件通信的几种方式详解

    前言 刚入门React可能会因为React的单向数据流的特性而遇到组件间沟通的麻烦,下面这篇文章就来给大家详细介绍下,在开始之前先来看一张图: react组件通信 需要组件之进行通信的几种情况 父组件向子组件通信 子组件向父组件通信 跨级组件通信 没有嵌套关系组件之间的通信 1. 父组件向子组件通信 React数据流动是单向的,父组件向子组件通信也是最常见的;父组件通过props向子组件传递需要的信息 Child.jsx import React from 'react'; import Pro

  • 浅谈React + Webpack 构建打包优化

    本文介绍了React + Webpack 构建打包优化,分享给大家,具体如下: 使用 babel-react-optimize对 React 代码进行优化 检查没有使用的库,去除 import 引用 按需打包所用的类库,比如 lodash . echart 等 lodash 可以采用babel-plugin-lodash进行优化. 需要注意的是 在 babel-react-optimize 中使用了 babel-plugin-transform-react-remove-prop-types 这

  • 浅谈React组件之性能优化

    高德纳: "我们应该忘记忽略很小的性能优化,可以说97%的情况下,过早的优化是万恶之源,而我们应该关心对性能影响最关键的另外3%的代码." 不要将性能优化的精力浪费在对整体性能提高不大的代码上,而对性能有关键影响的部分,优化并不嫌早.因为,对性能影响最关键的部分,往往涉及解决方案核心,决定整体的架构,将来要改变的时候牵扯更大. 1. 单个React组件的性能优化 React利用Virtual DOM来提升渲染性能,虽然每一次页面更新都是最组件的从新渲染,但是并不是将之前的渲染内容全部抛

  • 详解React中的todo-list

    基于React的一个简单Todo-list 先赌为快:在线DEMO,感觉还不错点一下star  -_- ~ 源码地址: 一.已经完成的功能 1.新增选项(默认未完成) 2.完成状态可以切换 3.当前选项可以删除 4.全部选项选中状态切换 5.全部个数,完成个数,未完成个数实时读取 6.刷新状态不变 7.双击可以编辑(有个坑:双击编辑内input的keyUp Enter保存会连带触发blur失去焦点保存.已解决:通过设置一个可以保存的状态控制) 二.待完成(新增路由) 三.目录结构 3.1.主要逻

  • React中获取数据的3种方法及优缺点

    为了保证的可读性,本文采用意译而非直译. 在执行 I/O 操作(例如数据提取)时,要先发送网络请求,然后等待响应,接着将响应数据保存到组件的状态,最后进行渲染. 在 React 中生命周期方法.Hooks和 Suspense是获取数据的方法.接下用事例演示一下如何使用它们并说明每种方法的优点和缺点,以便咱们更好的编写异步操作代码. 1.使用生命周期方法请求数据 应用程序Employees.org做两件事: 1.一进入程序就获取20名员工. 2.可以通过过滤条件来筛选员工. 在实现这两个需求之前,

  • 浅谈React中组件逻辑复用的那些事儿

    基本每个开发者都需要考虑逻辑复用的问题,否则你的项目中将充斥着大量的重复代码.那么 React 是怎么复用组件逻辑的呢?本文将一一介绍 React 复用组件逻辑的几种方法,希望你读完之后能够有所收获.如果你对这些内容已经非常清楚,那么略过本文即可. 我已尽量对文中的代码和内容进行了校验,但是因为自身知识水平限制,难免有错误,欢迎在评论区指正. 1. Mixins Mixins 事实上是 React.createClass 的产物了.当然,如果你曾经在低版本的 react 中使用过 Mixins,

  • 记一次react前端项目打包优化的方法

    前文 之前一年多前接手的一个react项目,前段时间因为做业务中台项目,对公司现有的应用项目做中台化改造,这期间将项目部署到uat环境,测试期间,测试小妹妹和产品大叔都吐槽进入uat项目的时候要load很久,白屏时间超过30s,体验很差,生产不至于这么慢但也是白屏时间挺长的,所以减少白屏时间增加用户体验成为了当务之急.复制代码 分析 通过控制台判断加载资源时间还有资源大小 通过开发者工具可以看到白屏的主要原因在于bundle.js这个打包后的文件过大,达到3.6M加上uat环境带宽等问题的话,光

  • React中Ref 的使用方法详解

    本文实例讲述了React中Ref 的使用方法.分享给大家供大家参考,具体如下: React中Ref 的使用 React v16.6.3 在典型的React数据流中,props是父组件与其子组件交互的唯一方式.要修改子项,请使用new props 重新呈现它.但是,在某些情况下,需要在典型数据流之外强制修改子项.要修改的子项可以是React组件的实例,也可以是DOM元素.对于这两种情况,React都提供了api. 何时使用refs refs有一些很好的用例: 1.文本选择或媒体播放. 2.触发势在

随机推荐