本文实例讲述了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.jb51.net/code/HtmlJsRun测试上述代码运行效果。
更多关于JavaScript相关内容可查看本站专题:《JavaScript常用函数技巧汇总》、《javascript面向对象入门教程》、《JavaScript错误与调试技巧总结》、《JavaScript数据结构与算法技巧总结》及《JavaScript数学运算用法总结》
希望本文所述对大家JavaScript程序设计有所帮助。
JS,Generator,函数
免责声明:本站文章均来自网站采集或用户投稿,网站不提供任何软件下载或自行开发的软件! 如有用户或公司发现本站内容信息存在侵权行为,请邮件告知! 858582#qq.com
《魔兽世界》大逃杀!60人新游玩模式《强袭风暴》3月21日上线
暴雪近日发布了《魔兽世界》10.2.6 更新内容,新游玩模式《强袭风暴》即将于3月21 日在亚服上线,届时玩家将前往阿拉希高地展开一场 60 人大逃杀对战。
艾泽拉斯的冒险者已经征服了艾泽拉斯的大地及遥远的彼岸。他们在对抗世界上最致命的敌人时展现出过人的手腕,并且成功阻止终结宇宙等级的威胁。当他们在为即将于《魔兽世界》资料片《地心之战》中来袭的萨拉塔斯势力做战斗准备时,他们还需要在熟悉的阿拉希高地面对一个全新的敌人──那就是彼此。在《巨龙崛起》10.2.6 更新的《强袭风暴》中,玩家将会进入一个全新的海盗主题大逃杀式限时活动,其中包含极高的风险和史诗级的奖励。
《强袭风暴》不是普通的战场,作为一个独立于主游戏之外的活动,玩家可以用大逃杀的风格来体验《魔兽世界》,不分职业、不分装备(除了你在赛局中捡到的),光是技巧和战略的强弱之分就能决定出谁才是能坚持到最后的赢家。本次活动将会开放单人和双人模式,玩家在加入海盗主题的预赛大厅区域前,可以从强袭风暴角色画面新增好友。游玩游戏将可以累计名望轨迹,《巨龙崛起》和《魔兽世界:巫妖王之怒 经典版》的玩家都可以获得奖励。