JS Generator 函数的含义与用法实例总结


Posted in Javascript onApril 08, 2020

本文实例讲述了JS Generator 函数的含义与用法。分享给大家供大家参考,具体如下:

读阮一峰老师《Generator 函数的含义与用法》总结

老师的文章通俗易懂,但是我个人理解上面有一些差,所以看了几遍之后才有呢么一点点体会

把它记录下来。

还是那句话,所有事物的出现都是为了解决对应的问题。

那么Generator出现是为了解决什么问题的呢?

在异步编程的场景下,如果有多个异步任务,如何处理他们的先后执行顺序?

举一个常见的例子,jquery的ajax请求,每一个success都是一个异步任务。

那么问题来了,如果我要保证页面渲染要在5个网络请求都回来之后再去渲染页面。

我们的代码就会变成:

$.ajax({...success:function(data){
  $.ajax({...success:function(data){
    $.ajax({...success:function(data){
      $.ajax({...success:function(data){
        $.ajax({...success:function(data){
          //do something
        }})
      }})
    }})
  }})
}})

这就是”回调函数噩梦”(callback hell)

为了解决这个问题,后来出现了Deferred和promise

两者区别不大,通过一种包装写法来减少回调函数

上面的ajax就可以写成:

ajax1 = $.ajax({...success:function(data){});
ajax2 = $.ajax({...success:function(data){});
ajax3 = $.ajax({...success:function(data){});
ajax4 = $.ajax({...success:function(data){});
ajax5 = $.ajax({...success:function(data){});
$.when(ajax1,ajax2,ajax3,ajax4,ajax5).done(function(
  //do something
)).then(function(){
  //do something2
})

1.8版本以上的jquery ajax模块默认返回Deferred对象

Deferred和promise将回调函数做拆分,将异步任务的处理和执行分成两部分完成

他们最大的问题就是代码冗余,包装之后的代码都需要通过then,done来执行后面的内容,也导致层次感不清晰

那有没有一种比较无感,简单的写法呢?

那就是协程,
我之前也是在这个地方困惑了很久,
前面说的大多日常用到过,也清楚一些原理,
关于协程用到的就少了,我们来分析下吧。

直接看一下协程的例子:

function asnycJob() {
 // ...其他代码
 var f = yield readFile(fileA);
 // ...其他代码
}

阮一峰老师的原话:

上面代码的函数 asyncJob 是一个协程,它的奥妙就在其中的 yield 命令。它表示执行到此处,执行权将交给其他协程。也就是说,yield命令是异步两个阶段的分界线。 协程遇到 yield 命令就暂停,等到执行权返回,再从暂停的地方继续往后执行。它的最大优点,就是代码的写法非常像同步操作,如果去除yield命令,简直一模一样。

之前没理解的原因就是没好好读这两句话,今认真看了一下,茅塞顿开。重要的有这么几点

首先asnycJob这个方法就是一个协程

yield相当于return,会返回当前程序的执行状态

当执行到yield,程序挂起等待返回后继续执行。

挂起这段时间去执行其他协程函数

Generator函数是ES6对协程函数的实现,

Generator函数的特点就是可以暂停代码执行。

跟协程函数一样,遇到yield关键字就暂停代码执行,

跟普通函数的区别在于Generator函数不会反悔结果,而是返回指针对象,

通过指针的next方法移动指针指向下一个yield关键字位置。

也就是说Generator函数的分阶段执行是由next方法控制的。

使用了Generator函数之后会对我们的代码有多大的改变呢?

fangction* gen(){
  var url = 'user/get/info';
  var data = yield $.get({url:url});
  console.log(data.userName);
}

你不需要担心远程接口的返回时机,完全按照同步的方式写代码就行。

但是也有缺点,Generator函数把一步操作做的很简洁,但对流程的管理却不方便,

上面的例子如何执行?

var g = gen();
g.next();
g.next();

next 方法的作用是分阶段执行 Generator 函数。每次调用 next 方法,会返回一个对象,

表示当前阶段的信息( value 属性和 done 属性)。value 属性是 yield 语句后面表达式的值,表示当前阶段的值;

done 属性是一个布尔值,表示 Generator 函数是否执行完毕,即是否还有下一个阶段。

你需要执行两次.next方法,来将你的Generator函数执行完毕。

关于如何自动化异步任务的流程管理,就需要co,thunk,async的帮助了

原文:Generator 函数的含义与用法

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

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

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

Javascript 相关文章推荐
提高 DHTML 页面性能
Dec 25 Javascript
如何获取JQUERY AJAX返回的JSON结果集实现代码
Dec 10 Javascript
浅谈JS继承_借用构造函数 & 组合式继承
Aug 16 Javascript
jQuery中ScrollTo用法示例
Sep 04 Javascript
js仿支付宝多方框输入支付密码效果
Sep 27 Javascript
javascript 单例模式详解及简单实例
Feb 14 Javascript
利用HBuilder打包前端开发webapp为apk的方法
Nov 13 Javascript
Bootstrap导航菜单点击后无法自动添加active的处理方法
Aug 10 Javascript
jQuery实现高级检索功能
May 28 jQuery
vue-cli脚手架引入弹出层layer插件的几种方法
Jun 24 Javascript
你或许不知道的一些npm实用技巧
Jul 04 Javascript
详解JVM系列之内存模型
Jun 10 Javascript
Vue列表循环从指定下标开始的多种解决方案
Apr 08 #Javascript
《javascript设计模式》学习笔记七:Javascript面向对象程序设计组合模式详解
Apr 08 #Javascript
vue开发移动端底部导航条功能
Apr 08 #Javascript
《javascript设计模式》学习笔记五:Javascript面向对象程序设计工厂模式实例分析
Apr 08 #Javascript
vue实现表单未编辑或未保存离开弹窗提示功能
Apr 08 #Javascript
JS快速实现简单计算器
Apr 08 #Javascript
javascript中contains是否包含功能实现代码(扩展字符、数组、dom)
Apr 07 #Javascript
You might like
用PHP+java实现自动新闻滚动窗口
2006/10/09 PHP
CI框架表单验证实例详解
2016/11/21 PHP
yii框架搜索分页modle写法
2016/12/19 PHP
用js获取电脑信息(是使用与IE浏览器)
2013/01/15 Javascript
jQuery Mobile页面跳转后未加载外部JS原因分析及解决
2013/03/18 Javascript
javascript判断chrome浏览器的方法
2014/03/26 Javascript
原生javascript实现的分页插件pagenav
2014/08/28 Javascript
jQuery实现仿Alipay支付宝首页全屏焦点图切换特效
2015/05/04 Javascript
jquery实现点击向下展开菜单项(伸缩导航)效果
2015/08/22 Javascript
AngularJS基础 ng-init 指令简单示例
2016/08/02 Javascript
AngularJS中的DOM操作用法分析
2016/11/04 Javascript
javascript解析ajax返回的xml和json格式数据实例详解
2017/01/05 Javascript
基于JS实现翻书效果的页面切换样式
2017/02/16 Javascript
微信小程序App生命周期详解
2018/01/31 Javascript
详解Angular操作cookies方法
2018/06/01 Javascript
JS使用正则表达式判断输入框失去焦点事件
2019/10/16 Javascript
vue-next/runtime-core 源码阅读指南详解
2019/10/25 Javascript
node.js express捕获全局异常的三种方法实例分析
2019/12/27 Javascript
解决vux 中popup 组件Mask 遮罩在最上层的问题
2020/11/03 Javascript
Python如何读取MySQL数据库表数据
2017/03/11 Python
Python的SimpleHTTPServer模块用处及使用方法简介
2018/01/22 Python
Python pymongo模块用法示例
2018/03/31 Python
解决Python下json.loads()中文字符出错的问题
2018/12/19 Python
将python运行结果保存至本地文件中的示例讲解
2019/07/11 Python
python获取指定日期范围内的每一天,每个月,每季度的方法
2019/08/08 Python
在python3中使用shuffle函数要注意的地方
2020/02/28 Python
为2021年的第一场雪锦上添花:用matplotlib绘制雪花和雪景
2021/01/05 Python
全方位了解CSS3的Regions扩展
2015/08/07 HTML / CSS
使用canvas实现黑客帝国数字雨效果
2020/01/02 HTML / CSS
英国家居用品和床上用品零售商:P&B Home
2020/01/16 全球购物
导游词书写之黄山
2019/08/06 职场文书
Vue h函数的使用详解
2022/02/18 Vue.js
SQL Server查询某个字段在哪些表中存在
2022/03/03 SQL Server
基于Apache Hudi在Google云构建数据湖平台的思路详解
2022/04/07 Servers
分享很少见很有用的SQL功能CORRESPONDING
2022/08/05 MySQL
Go gorilla/sessions库安装使用
2022/08/14 Golang