ES6 Generator函數(shù)的應用實例分析
本文實例講述了ES6 Generator函數(shù)的應用。分享給大家供大家參考,具體如下:
Generator 函數(shù)是 一種異步編程解決方案,Generator 函數(shù)會返回一個遍歷器對象,Generator 函數(shù)是一個普通函數(shù),但是有兩個特征。一是,function關(guān)鍵字與函數(shù)名之間有一個星號;二是,函數(shù)體內(nèi)部使用yield表達式。
認識generator函數(shù)
function * fn() {
console.log('hello');
return 'Joh';
}
// 執(zhí)行fn()時不會直接執(zhí)行方法體中的代碼,它會返回一個指針, 這個指針實現(xiàn)了 interator接口,也就是返回一個interator對象
let it = fn();
// 通過調(diào)用next就會執(zhí)行方法體, 返回結(jié)果是 {value:'Joh', done:true}, 其中next返回的是函數(shù)體中return的值
let res = it.next();
console.log(res);
generator函數(shù)中的yield 與 return
function * fn() {
// 和yield相配合,把一個generator內(nèi)部分為幾個斷點來執(zhí)行,每個斷點就是yield語句
// 注意 yield和return的區(qū)別: yield可以有多個,return只能有1個
yield 1;
yield 2;
yield 3;
return 4; // 可以沒有return值,done為true的value將會是undefined
}
let it = fn();
// 在for-of 循環(huán)中只能打印done為false的value值,done為true時,程序終止
for(let v of it) {
console.log(v); // 分別輸出 1 2 3
}
yield 的值與賦值語句
function * fn(_name) {
let name = yield _name; // yield 的默認值為undefined
return name;
}
let it = fn('Joh');
console.log(it.next()); // {value:'Joh', done:false}
console.log(it.next('Tom')); // {value:'Tom', done:true} // 此處value應該為undefined,但是通過next參數(shù)的形式賦值改變了最后一個值
console.log(it.next('Lily')); // {value: undefined, done:true} // 已經(jīng)循環(huán)完畢,即使傳值也是undefined
yield 語句的位置與括號
function sum(a, b) {
return a + b;
}
function * fn() {
let res = sum(yield 1, 5 + (yield 3));
console.log(res);
console.log('my qq: ' + (yield qq)); // yield 在一個語句中需要括起來
}
fn();
yield 異常捕獲
異常捕獲的方式1:
function * fn() {
let qq = yield; // yield 默認返回undefined, 不會拋出異常
console.log(qq);
}
let g = fn();
g.next(); // 第一個斷點沒有輸出
// g.next('qq 11111'); // 完畢之后傳值輸出:qq 11111
g.throw('error!'); // Uncaught error!
異常捕獲的方式2:
function * fn() {
let qq;
try {
qq = yield; // yield 默認返回undefined
}catch(e){
console.log('qq have error');
}finally{
console.log(qq);
}
}
let g = fn();
g.next();
g.throw('error!');
// qq have error
// undefined
異常捕獲的方式3:
function * fn() {
let qq;
qq = yield;
console.log(qq);
}
let g = fn();
g.next();
try{
g.throw('error!');
}catch(e){
console.log('qq have error!');
}
異常捕獲的方式4:
function * fn() {
let qq;
try {
qq = yield ff; // ff 未定義, 所以qq不會被正確賦值 此處是非 yield 的異常
}catch(e){
console.log('err1');
}
console.log(qq);
}
let g = fn();
g.next();
g.next('qq 5554');
// err1
// undefined
利用generator和promise結(jié)合使用,讓異步的邏輯關(guān)系,使用同步的方式書寫
function asyncF(name) {
return new Promise(function(resolve){
setTimeout(function(){
resolve('my name is ' + name);
});
});
}
function * fn() {
console.log(yield asyncF('Joh'));
}
let gf = fn();
function exec(gf,value) {
let res = gf.next(value);
if(!res.done) {
if(res.value instanceof Promise) {
res.value.then(function (v) {
exec(gf, v);
})
}else{
exec(gf, res.value);
}
}
}
exec(gf); // my name is Joh
更復雜的寫法:
function asyncF(name) {
return new Promise(function(resolve){
setTimeout(function(){
resolve('my name is ' + name);
});
});
}
function sum(a, b) {
return new Promise(function (resolve) {
setTimeout(function () {
resolve(a + b);
});
})
}
function * fn(name) {
if((yield sum(3,5)) > 6) {
console.log(yield asyncF(name));
}else{
console.log('error');
}
}
let gf = fn('Joh');
// generator 執(zhí)行器 相當于 tj/co 模塊
function exec(gf,value) {
let res = gf.next(value);
if(!res.done) {
if(res.value instanceof Promise) {
res.value.then(function (v) {
exec(gf, v);
})
}else{
exec(gf, res.value);
}
}
}
exec(gf); // my name is Joh
使用純promise實現(xiàn):
function asyncF(name) {
return new Promise(function(resolve){
setTimeout(function(){
resolve('my name is ' + name);
});
});
}
function sum(a, b) {
return new Promise(function (resolve) {
setTimeout(function () {
resolve(a + b);
});
})
}
function fn(name) {
sum(3,5)
.then(function (num) {
if(num > 6) {
asyncF(name)
.then(function (v) {
console.log(v);
})
}else{
console.log('error');
}
})
}
fn('Joh');
使用co模塊,來代替自己寫的執(zhí)行器
var co = require('co');
function asyncF(name) {
return new Promise(function(resolve){
setTimeout(function(){
resolve('my name is ' + name);
});
});
}
function sum(a, b) {
return new Promise(function (resolve) {
setTimeout(function () {
resolve(a + b);
});
})
}
function * fn(name) {
if((yield sum(3,5)) > 6) {
console.log(yield asyncF(name));
}else{
console.log('error');
}
}
var fnx = co.wrap(fn);
fnx('Joh'); // my name is Joh
更多關(guān)于JavaScript相關(guān)內(nèi)容可查看本站專題:《javascript面向?qū)ο笕腴T教程》、《JavaScript切換特效與技巧總結(jié)》、《JavaScript查找算法技巧總結(jié)》、《JavaScript錯誤與調(diào)試技巧總結(jié)》、《JavaScript數(shù)據(jù)結(jié)構(gòu)與算法技巧總結(jié)》、《JavaScript遍歷算法與技巧總結(jié)》及《JavaScript數(shù)學運算用法總結(jié)》
希望本文所述對大家JavaScript程序設(shè)計有所幫助。
相關(guān)文章
javascript實現(xiàn)判斷鼠標的狀態(tài)
這篇文章主要介紹了javascript實現(xiàn)判斷鼠標的狀態(tài)的相關(guān)代碼,十分簡單實用,有需要的小伙伴可以參考下。2015-07-07
JavaScript中Promise的執(zhí)行順序詳解
Promise 是 JS 中進行異步編程的新的解決方案(舊的是純回調(diào)形式) ,下面這篇文章主要給大家介紹了關(guān)于JavaScript中Promise執(zhí)行順序的相關(guān)資料,文中通過實例代碼介紹的非常詳細,需要的朋友可以參考下2022-01-01
微信小程序開發(fā)(二):頁面跳轉(zhuǎn)并傳參操作示例
這篇文章主要介紹了微信小程序開發(fā)頁面跳轉(zhuǎn)并傳參操作,結(jié)合實例形式詳細分析了微信小程序頁面跳轉(zhuǎn)并傳參相關(guān)操作技巧,需要的朋友可以參考下2020-06-06
JavaScript中setInterval的用法總結(jié)
這篇文章主要是對JavaScript中setInterval的用法進行了詳細的總結(jié)介紹,需要的朋友可以過來參考下,希望對大家有所幫助2013-11-11
JS實現(xiàn)經(jīng)典的中國地區(qū)三級聯(lián)動下拉菜單功能實例【測試可用】
這篇文章主要介紹了JS實現(xiàn)經(jīng)典的中國地區(qū)三級聯(lián)動下拉菜單功能,結(jié)合完整實例形式分析了javascript基于事件響應實現(xiàn)頁面元素動態(tài)變換的相關(guān)操作技巧,需要的朋友可以參考下2017-06-06
JavaScript檢測并限制復選框選中個數(shù)的方法
這篇文章主要介紹了JavaScript檢測并限制復選框選中個數(shù)的方法,涉及javascript針對復選框的判定與運算相關(guān)技巧,非常簡單實用,需要的朋友可以參考下2015-08-08
JS賦值、淺拷貝和深拷貝(數(shù)組和對象的深淺拷貝)實例詳解
這篇文章主要介紹了JS賦值、淺拷貝和深拷貝,結(jié)合實例形式總結(jié)分析了JavaScript數(shù)組和對象的深淺拷貝相關(guān)概念、原理、操作技巧與使用注意事項,需要的朋友可以參考下2020-03-03

