javascript 函数的暂停和恢复实例详解


Posted in Javascript onApril 25, 2020

本文实例讲述了javascript 函数的暂停和恢复。分享给大家供大家参考,具体如下:

javascript 异步编程从来都是一个难题,最开始我们用 callback,但随之触发了回调地狱,于是“发明” Promise 解决 callback 嵌套过深的问题。然而由于滥用 Promise(一连串的 then),代码变得不易阅读了。此时,async-await 横空出世,它让我们可以用同步的方式编写异步代码,简直 amazing,以至于有人说它就是 javascript 异步编程的银弹。

P.S.代码只是演示,并不可用

function getProfile(id) {
 return window.fetch(`https://api.com/wedding/profile/${weddingId}`
}

async function getWeddingDetail(weddingId) {
 try {
  // 暂停执行
  const wedding = await window.fetch(`https://api.com/wedding/${weddingId}`);
  // 当结果返回恢复执行,接着继续暂停
  const groom = await getProfile(wedding.groomId);
  // ... 恢复执行 -> 暂停 ...
  const bride = await getProfile(wedding.brideId);
  // ... 恢复执行
  return { wedding, bride, groom };
 } catch (error) {
  handleError(error);
 }
}

没有银弹

然而计算机科学领域中并不存在银弹。async-await 也有缺点,比如你忘了写 await,不信?

假设别人编写了一个工具函数叫getProfile,如果不了解它的具体实现,你是不是就把它当做同步函数,即便getProfile是异步的。

当然,这只是一个小问题,更让我难受的是,如果你在一个函数中使用了 async,那么调用它的函数也得变成一个 async,若还有另一个函数要调用这个调用函数......holly shit!现在你明白了吧。

有没有两全其美的办法?

// getWeddingDetail 根本不用关心内部的函数是异步or同步
function getWeddingDetail(weddingId) {
 const wedding = window.fetch(`https://api.com/wedding/${weddingId}`);
 const groom = getProfile(wedding.groomId);
 const bride = getProfile(wedding.brideId);
 return { wedding, bride, groom };
}

没有什么是一个中间层解决不了的

异步编程的核心,就是函数暂停和恢复执行。而决定一个函数是暂停还是恢复执行,这是 js 运行时干的活儿,难不成我们今天要深入引擎实现?

No!我不了解 C++,也不懂 js 引擎是如何实现的。

但是呢,我可以写一个中间层(函数runtime),尝试实现上面的需求,当然,这会有一些限制。

一、入口函数

假设要运行的函数如下:

function main() {
 const id = 123;
 console.log('Getting wedding:', id);

 const { wedding, bride, groom } = getWeddingDetail(id);

 console.log('Wedding detail:', wedding);
}

我们期望能够按照下面的方式运行:

function runtime(mainFn) {
 mainFn();
}

// start runtime
runtime(main);

基本框架已经有了,接着干啥?

首先,要搞清楚在不用 await 的前提下,如何中断函数运行。

然后,在合适的地方恢复执行。

js 中有两种方法中断函数运行:return和throw。我选择 throw,因为它表示遭遇异常导致的中断。好了,我们改造一下 runtime

function runtime(mainFn) {
 const _originalFetch = window.fetch;
 window.fetch = (url, options) => {
  // "暂停"
  throw new Error();
 };
 // 运行入口函数
 runMain();

 function runMain() {
  try {
   mainFn();
  } catch (error) {
   // 函数 "暂停"
   // 恢复并重新执行 mainFn
   runMain();
  }
 }
}

先忽略这段代码出现的问题,把目光聚集在函数“中断”“恢复”这两个点上,显然,目的已经达到。接下来对它进行优化。

首当其冲的是 runMain,只需要当 window.fetch 成功后再执行:

function runtime(mainFn) {
 const _originalFetch = window.fetch
 window.fetch = (url, options) => {
  _originalFetch(url, options).then(res => {
   // 返回结果后恢复执行
   runMain()
  })
  throw new Error()
 }

 runMain();

 function runMain() {
  try {
   mainFn();
  } catch (error) {
   // ignore
  }
 }
}

window.fetch 每次抛出异常,这导致 mainFn 无限循环的执行。

要解决这个问题,需要引入缓存,使得我们仅需要在第一次 fetch 时抛出异常,而为后面的请求返回响应。

function runtime(mainFn) {
 const _originalFetch = window.fetch
 windo.fetch = (url, options) => {
  if (cache.has([url, options])) return cache.get([url, options])

  _originalFetch(url, options).then(res => {
   cache.set([url, options], res)
   runMain()
  })

  throw new Error()
 }

 runMain();

 function runMain() {
  try {
   mainFn();
  } catch (error) {
   // ignore
  }
 }
}

成功啦!

运行程序,检查 console 的输出,由于重复运行了多次,'Getting wedding:', 123也被显示了多次,这是 console.log 的副作用导致的。

二、纯函数

runtime 只允许运行纯函数,如果你的代码中有副作用,则必须添加限制条件:runSideEffects().

function main() {
 const id = 123;
 runSideEffects(() => console.log('Getting wedding:', id));

 const { wedding, bride, groom } = getWeddingDetail(id);

 runSideEffects(() => console.log('Wedding detail:', wedding));
}

sideEffects 的实现非常容易:

function runtime(mainFn) {
 // 参考上面的代码

 // 提供 `runSideEffects`
 const sideEffects = [];
 window.runSideEffects = fn => {
  sideEffects.push(fn);
 };

 runMain();

 function runMain() {
  try {
   mainFn();
   sideEffects.forEach(fn => fn());
  } catch (error) {
   // 清除副作用
   sideEffects.splice(0, sideEffects.length);
  }
 }
 
}

再次运行,'Getting wedding:', 123只显示一次啦~

到底干了些啥?

为了模仿函数暂停和恢复,我们通过 throw 一个错误来“暂停”函数,重新运行来“恢复”函数。

为了从暂停处“恢复”,需要将抛出的错误替换成函数返回值,我们用缓存机制达到了这个目的。

最后,为了能安全的重复执行函数,需要将它转化为一个纯函数。如果有副作用,则将它们收集起来,在函数运行成功后,再执行副作用。

扯这么多,有什么实际用途?

本文的灵感来自于React Suspense。有了 Suspense,就可以像下面这样来获取数据:

function Component() {
 const data = getDataFromNetwork();
 return <div />;
}

getDataFromNetwork 将发起异步请求,所以它是一个异步函数,但 React 让它看来是是一个同步操作。这很有趣~

原文阅读:pause-and-resume-a-javascript-function

感兴趣的朋友可以使用在线HTML/CSS/JavaScript代码运行工具:http://tools.3water.com/code/HtmlJsRun测试上述代码运行效果。

更多关于JavaScript相关内容可查看本站专题:《JavaScript常用函数技巧汇总》、《javascript面向对象入门教程》、《JavaScript错误与调试技巧总结》、《JavaScript数据结构与算法技巧总结》及《JavaScript数学运算用法总结》

希望本文所述对大家JavaScript程序设计有所帮助。

Javascript 相关文章推荐
基于JQuery的数字改变的动画效果--可用来做计数器
Aug 11 Javascript
javascript中的作用域scope介绍
Dec 28 Javascript
js如何实现淡入淡出效果
Nov 18 Javascript
js移动焦点到最后位置的简单方法
Nov 25 Javascript
原生JS下拉加载插件分享
Dec 26 Javascript
jQuery+HTML5实现弹出创意搜索框层
Dec 29 Javascript
详解如何在Angular中快速定位DOM元素
May 17 Javascript
vue-cli与webpack处理静态资源的方法及webpack打包的坑
May 15 Javascript
JavaScript中filter的用法实例分析
Feb 27 Javascript
jquery图片预览插件实现方法详解
Jul 18 jQuery
react实现antd线上主题动态切换功能
Aug 12 Javascript
解决vue 子组件修改父组件传来的props值报错问题
Nov 09 Javascript
详解ES6 CLASS在微信小程序中的应用实例
Apr 24 #Javascript
Vue中函数防抖节流的理解及应用实现
Apr 24 #Javascript
vue 路由懒加载中给 Webpack Chunks 命名的方法
Apr 24 #Javascript
vue中使用带隐藏文本信息的图片、图片水印的方法
Apr 24 #Javascript
基于Vue实现微前端的示例代码
Apr 24 #Javascript
Vue通过getAction的finally来最大程度避免影响主数据呈现问题
Apr 24 #Javascript
小程序开发之模态框组件封装
Apr 23 #Javascript
You might like
PHP4实际应用经验篇(7)
2006/10/09 PHP
php将fileterms函数返回的结果变成可读的形式
2011/04/21 PHP
php获取远程文件大小
2015/10/20 PHP
Mootools 1.2教程 输入过滤第二部分(字符串)
2009/09/15 Javascript
jQuery ui 利用 datepicker插件实现开始日期(minDate)和结束日期(maxDate)
2014/05/22 Javascript
Javascript获取表单名称(name)的方法
2015/04/02 Javascript
在JavaScript中操作数组之map()方法的使用
2015/06/09 Javascript
javascript格式化日期时间方法汇总
2015/06/19 Javascript
js实现文本框只允许输入数字并限制数字大小的方法
2015/08/19 Javascript
详解JavaScript中的事件流和事件处理程序
2016/05/20 Javascript
js匿名函数作为函数参数详解
2016/06/01 Javascript
简单实现js间歇或无缝滚动效果
2016/06/29 Javascript
轻松掌握JavaScript策略模式
2016/08/25 Javascript
JS简单实现禁止访问某个页面的方法
2016/09/13 Javascript
jQuery中的siblings()是什么意思(推荐)
2016/12/29 Javascript
Node.js连接mongodb实例代码
2017/06/06 Javascript
VSCode中如何利用d.ts文件进行js智能提示
2018/04/13 Javascript
jQuery利用FormData上传文件实现批量上传
2018/12/04 jQuery
Vue组件通信的几种实现方法
2019/04/25 Javascript
Vue路由之JWT身份认证的实现方法
2019/08/26 Javascript
layui-table获得当前行的上/下一行数据的例子
2019/09/24 Javascript
vue中在vuex的actions中请求数据实例
2019/11/08 Javascript
JS函数本身的作用域实例分析
2020/03/16 Javascript
[48:31]完美世界DOTA2联赛PWL S3 DLG vs Phoenix 第二场 12.17
2020/12/19 DOTA
总结Python编程中三条常用的技巧
2015/05/11 Python
python常用排序算法的实现代码
2019/11/08 Python
Python高级特性——详解多维数组切片(Slice)
2019/11/26 Python
HTML5 本地存储和内容按需加载的思路和方法
2011/04/07 HTML / CSS
英国殿堂级有机护肤品牌:Rodial
2017/04/17 全球购物
俄罗斯天然和有机产品、健康生活网上商店:Fitomarket.ru
2020/10/09 全球购物
表彰大会策划方案
2014/05/13 职场文书
艺术设计专业求职自荐信
2014/05/19 职场文书
工作证明格式及范本
2014/09/12 职场文书
诚信高考倡议书
2019/06/24 职场文书
超详细教你怎么升级Mysql的版本
2021/05/19 MySQL
Golang 字符串的常见操作
2022/04/19 Golang