文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

为什么 Redux-Saga 不能用 Async Await 实现

2024-12-02 20:11

关注

今天群里有个小伙伴问了个问题

为什么 saga 不能用 async await 来实现呢?

想必开始接触 redux-saga 的同学都有这个疑问,为啥为要用 generator 的写法,用 async await 行不行。

  1. import { put, call } from 'redux-saga/effects' 
  2. import { loginService } from '../service/request' 
  3.  
  4. function* login(action) { 
  5.     try { 
  6.         const loginInfo = yield call(loginService, action.account) 
  7.         yield put({ type: 'loginSuccess', loginInfo }) 
  8.     } catch (error) { 
  9.         yield put({ type: 'loginFail', error }) 
  10.     } 

这个问题我刚开始用 saga 的时候也想问,但后来了解了 saga 的原理就想明白了。

下面我们就来探究一下。

saga 原理

我们从组件把 action 发给 store,这个过程是同步的。

但是有一些异步的过程加在哪里呢?中间件。

redux saga 会先把 action 直接透传给 store,这个过程是同步的。

然后再传递一份给 watcher saga,看一下是否是被监听的 action,如果是交给 worker saga 来处理,worker saga 处理的过程中可以 put 新的 action 到 store,这个过程是异步的。

这就是 redux-saga 的原理,原理上还是比较简单的,亮点在于异步过程的组织上,也就是 generator 上。

为什么说用 generator 来组织异步过程是 redux-saga 的亮点呢?

别急,我们先了解下什么是 generator。

generator

生成器(generator)是一个产生迭代器(iterator)的函数,通过 yield 返回一个个值。

而迭代器是一个有 value 和 done 属性的对象,用于顺序遍历某个集合。

我们可以调用 iterator.next 取到 yield 生成的一个个值。

也可以用 Array.from 或者展开运算符来取,这是 iterator 的特点。

除了 next 方法,迭代器还有 return 和 throw 方法,就像函数里的 return 和 throw 语句一样。

比如用 iterator.return 中止后续流程

用 iterator.throw 抛出错误

也就是说 generator 的执行是要由一个执行器来控制的,什么时候取下一个 yield 出的值,什么时候 next,什么时候 return 什么时候 throw 都是由执行器控制。

执行器可以通过 next、return、throw 的参数传递给 generator 执行后的结果:

上面这段代码就是一个小型 saga 了,原理就这么简单。

那为什么 async await 不行呢?

async await

当 generator 返回的值都是 Promise,那么执行 Promise 以后,只有 resolve 和 reject 两种结果,这个执行器就很固定,那自然可以写一个通用的执行器来自动调用 next、throw 和 return。

这个就是 async await 的原理,只不过被做成了语法糖。

async await 本质上不过是一个 generator 的执行器。

如果 redux-saga 用 async await 实现,那么所有的异步逻辑都要命令式的写在 await 后面,这样会导致异步过程很难测试。所以 redux-saga 自己实现了一个执行器。

再来看这段 saga 的代码:

  1. import { put, call } from 'redux-saga/effects' 
  2. import { loginService } from '../service/request' 
  3.  
  4. function* login(action) { 
  5.     try { 
  6.         const loginInfo = yield call(loginService, action.account) 
  7.         yield put({ type: 'loginSuccess', loginInfo }) 
  8.     } catch (error) { 
  9.         yield put({ type: 'loginFail', error }) 
  10.     } 

generator 中 yield 出的不是 promise,而是一个个 effect,这个其实就是一个对象,声明式的告诉 saga 执行器要做什么,而不是命令式的自己实现。

这样 generator 的执行器就根据不同的 effect 做不同的实现:

call effect 有对应的实现、put effect 也有对应的实现,也就是说 saga 的 generator 执行器不像 async await 那样什么都不做,而是有自己的 runtime 在的。

这样有什么好处呢?

现在,我们回到最开始那个问题,redux-saga 能用 async await 实现么?

能,但是 async await 是一个 generator 的自动执行器,没有 runtime 逻辑,要命令式的把异步过程写在 saga 里。如果自己实现一个 generator 执行器,那么就可以把异步过程抽离出来,方便组织、方便测试。用 async await 实现 saga 的话,那就失去了灵魂。

总结

redux-saga 的原理是透传 action 到 store,然后再传一份 aciton 到 saga 组织的异步过程,saga 分为 watcher saga 和 worker saga。watcher saga 判断 action 是否要处理,然后交给 wroker saga 处理。

生成器 generator 是返回迭代器 iterator 的函数,iterator 有 next、throw、return 等方法,需要配合一个执行器来执行。

async await 本质上就是一个 generator 的自动执行器。

用 async await 实现 redux saga 的话,那就要开发者命令式的组织异步过程,还难以测试。

所以 redux-saga 自己实现了一个 generator 执行器,自带 runtime。generator 只要返回 effect 对象来声明式的说明要执行什么逻辑,然后由相应的 effect 实现来执行。

这种声明式的思路除了易于组织异步过程外,还有非常好的可测试性。

 

generator + saga 执行器的设计是 redux-saga 的灵魂,所以不能用 async await 来实现。

 

来源:神光的编程秘籍内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯