生成器的应用 async与await实现

news2025/3/31 17:08:42

生成器配合使用函数

  • yield 将暂停执行代码,同时把函数返回值传递出去
function s(){
    console.log('ss');
}
function * f(){

    /*
        当 next() 调用时
        从头开始执行
        直到yield 开始检查后面的表达式
        现在是一个函数,那么首先执行函数
        当函数执行完毕,有返回值

        下面相当于 yield undefined
    */
    yield s();
}

//生成器对象
let g = f();    
//开始执行
console.log(g.next());//{value: undefined, done: false}

生成器配合使用Promise

  • 第一步
  • yield 可以把返回值传递出去,由next()来接受

function do_it(arg){
    return new Promise((r , j ) =>{
        setTimeout(() => {
            r(arg);
        }, 1000);
    });
}


function * f(){
    //yield new Promise(...)
    yield do_it('第一步');
}

let g = f();
// res :{value: Promise, done: false}
// res.value 是 yield 传回来的Promise
let res = g.next();
res.value.then(res=>{
    //第一步
    console.log(res);
});
  • 第二步
  • next(参数) 可以把参数传递给yield,让yield恢复执行
function do_it(arg){
    return new Promise((r , j ) =>{
        setTimeout(() => {
            r(arg);
        }, 1000);
    });
}


function * f(){
    /*
        yield Promise 
        并暂停流程
    */
    let res = yield do_it('第一步');

    //第一次: 第一步
    console.log('第一次:',res);

    // yield 第二个 Promise, 同时暂停执行
    res = yield do_it(res + ' 第二步');
    
}

let g = f();
// next 获得 yield 的第一个Promise
let res = g.next();
res.value.then(res=>{
    
    /*
        等第一个Promise执行完后
        next将第一个Promise的结果传递进去
        并获得第二个Promise
    */
    let r = g.next(res);   

    // 第二个Promise
    r.value.then(res=>{
        //第一步 第二步
        console.log(res);
    });
});
  • 第三步,改良上面的代码
  • 上面代码中都需要自己手写then,再调用next()来执行,其代码结构是一致的
  • 利用递归自动完成

function do_it(arg){
    return new Promise((r , j ) =>{
        setTimeout(() => {
            r(arg);
        }, 1000);
    });
}


function * f(){
    let res = yield do_it('第一步');
    console.log(res);   //第一步
    res = yield do_it(res + ' 第二步');
    console.log(res);   //第一步 第二步
    res = yield do_it(res + ' 第三步');
    console.log(res);   //第一步 第二步 第三步
}


//使用递归来自动化完成
function auto_run(g){

    function run(arg){
        let result = g.next(arg);
        if(result.done)
            return;
        result.value.then(res=>{
            run(res);
        });
    }
    
    run();
}

auto_run(f());

进化到async await

  • 使用async await来简化上面操作
  • 可先看后面的async await的使用

function do_it(arg){
    return new Promise((r , j ) =>{
        setTimeout(() => {
            r(arg);
        }, 1000);
    });
}

// 异步迭代器 async function *
async function* f(){
    let res = await do_it('第一步');
    console.log(res);
    res = await do_it(res + ' 第二步');
    console.log(res);
    res = await do_it(res + ' 第三步');
    console.log(res);
}


(async ()=>{
    let g = f();
    let r = await g.next(); //一直迭代到{done:true}时
    console.log('awwait:',r);
})();


async

  • async 返回一个Promise
async function f(){
    //return Promise.resolve(123);
    return 123;
}
let pr = f();
pr.then(res=>{
    console.log(res);   //123
});

  • 如果函数内部有异常则返回一个Promise.reject();
async function f(){
    throw new Error('error');
}
let pr = f();
//增加一个 rejected 回调
pr.then(null,res=>{
    console.log(res);   //error
});

async声明的函数

  • 以下两者等价
async function f(){
    return 123;
}

function ff(){
    return new Promise((resolve, reject )=>{
        try{
            resolve(123);
        }catch(e){
            reject(e);
        }
    });
}

await

  • await后跟一个表达式,需要在async声明的函数中使用
  • yield的增强版, await将先执行表达式,再暂停流程
  • yield需要手动调用next()执行下一步, await全自动
  • await将创建一个Promise : 使用Promise.resolve()
  • 简单理解就是使用then
async function f(){
    // await Promise.resolve(3);
    let res = await 3;
    console.log(res);  //3
}
f();
  • await 做了什么
async function f(){
    let result = await 3;
}

//相当于
async function f(){
    Promise.resolve(3).then(res=>{
        let result = res;
    });
}
  • 如果await 后面的表达式是一个Promise
//用于返回一个Promise
function test(){
    return new Promise((resolve , reject)=>{
        resolve(1);
    });
}

async function f(){
    let result = await test();
}
  • await相当于
function ff(){

    //执行此函数,返回一个Promise
    let ret = test(); 

    /*
        await创建一个Promise
        使用resolve把test()返回值传入
    */

    let await_new_promise = new Promise((resolve,reject)=>{
        /*
            resolve的调用时机在: ret 中的 resolve/reject 调用后
            let res = ret.resolve(1);
            resolve(res);
        */
        resolve(ret);
    });

    // 使用then添加注册函数来获取值
    await_new_promise.then(res=>{

        // 相当于 let result = await test();
        let result = res;

        console.log(res);   //1
    });
}

await的yield特性体现在哪里

  • 把上面的代码稍作修改
  • 把await的代码放入一个生成器中,加入yield
  • 外部代码就能使用 .next() 来控制流程
function * gen(){

    //先调用test(),返回的是一个Promise
    let ret = test();

    //创建一个Promise
    let await_new_promise = new Promise((await_resolve, await_reject)=>{

        //使用Promise.resolve来处理结果
        await_resolve(ret);
    });

    // 传递一个Promise
    yield await_new_promise;
}

async await 整体实现

  • 根据上面所知 async 声明的函数将返回一个Promise
  • await将创建一个Promise,并使用then来获取值
  • await是增强版的yield,可暂停流程,也就是使用生成器来控制流程
  • 有如下代码
function test(){
    return new Promise((resolve , reject)=>{
        resolve(1);
    });
}

async function f(){
    let result = await test();
}
  • 异步函数 f 的完整实现
function ff(){

// async 声明的函数将返回一个Promise

return new Promise((r , j)=>{

    //await 做的事
    function * gen(){

        let ret = test();
        let await_new_promise = new Promise((await_resolve, await_reject)=>{
            await_resolve(ret);
        });
        
        yield await_new_promise;
    }

    // 外部控制流程
    try {

        // 第一个 await 
        let g = gen();
        
        //next返回一个对象{done:true/false,value:yield后跟的值}
        let obj = g.next();

        //获取Promise
        let first_await = obj.value;

        //最终结果
        let result = undefined;

        //then中添加fulfilled对应回调来获取结果
        first_await.then(res=>{
            result = res;

            console.log(res); // 1
        });

        r(result);
    } catch (error) {
        j(error);
    }
});
}
ff();

与生成器代码比较


function do_it(arg){
    return new Promise((r , j ) =>{
        setTimeout(() => {
            r(arg);
        }, 1000);
    });
}

// 生成器
function * f(){
    let res = yield do_it('第一步');
    console.log(res);   //第一步
    res = yield do_it(res + ' 第二步');
    console.log(res);   //第一步 第二步
    res = yield do_it(res + ' 第三步');
    console.log(res);   //第一步 第二步 第三步
}
/*
    生成器需要自己手动调用
    或者写递归函数来自动调用

    let g = f();
    let res = g.next();
    res.value.then(res=>{
        g.next(res);
        ....
        g.next()...
    });
*/

//------------------------------------------------

/*
    使用async await
    全自动
*/
async function ff(){
    let res = await do_it(1);
    res = await do_it(res+1);
    res = await do_it(res+1);
    console.log('res:',res); // 3
}
ff();

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2323524.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Apache Shiro 统一化实现多端登录(PC端移动端)

Apache Shiro 是一个强大且易用的Java安全框架,提供了身份验证、授权、密码学和会话管理等功能。它被广泛用于保护各种类型的应用程序,包括Web应用、桌面应用、RESTful服务、移动端应用和大型企业级应用。 需求背景 在当今数字化浪潮的推动下&#xff…

NAT—地址转换(实战篇)

一、实验拓扑: 二、实验需求: 1.实现内网主机访问外网 2.实现外网客户端能够访问内网服务器 三、实验思路 1.配置NAT地址池实现内网地址转换成公网地址,实现内网主机能够访问外网。 2.配置NAT Sever实现公网地址映射内网服务器地址&…

用HTML和CSS生成炫光动画卡片

这个效果结合了渐变、旋转和悬浮效果的炫酷动画示例&#xff0c;使用HTML和CSS实现。 一、效果 二、实现 代码如下&#xff1a; <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport&quo…

FPGA_YOLO(三)

上一篇讲的是完全映射&#xff0c;也就是block中的所包含的所有的卷积以及归一&#xff0c;池化卷积 举例总共6个等都在pl侧进行处理&#xff08;写一个top 顶层 里面conv 1 bn1 relu1 pool1 conv1*1 conv 2 bn2 relu2 pool2 conv1*1 ....总共6个 &#xff09;&#xff0c;…

旅游CMS选型:WordPress、Joomla与Drupal对比

内容概要 在旅游行业数字化转型进程中&#xff0c;内容管理系统&#xff08;CMS&#xff09;的选择直接影响网站运营效率与用户体验。WordPress、Joomla和Drupal作为全球主流的开源CMS平台&#xff0c;其功能特性与行业适配性存在显著差异。本文将从旅游企业核心需求出发&…

全面适配iOS 18.4!通付盾加固产品全面升级,护航App安全上架

引言&#xff1a; 苹果官方新规落地&#xff01; 自2025年4月24日起&#xff0c;所有提交至App Store Connect的应用必须使用Xcode 16或更高版本构建&#xff0c;否则将面临审核驳回风险&#xff01;Beta版iOS 18.4、iPadOS 18.4现已推出&#xff0c;通付盾iOS加固产品率先完成…

一台电脑最多能接几个硬盘?

在使用电脑时&#xff0c;硬盘空间不够是许多用户都会遇到的问题。无论是摄影师、剪辑师等需要大量存储空间的专业人士&#xff0c;还是游戏玩家、数据备份爱好者&#xff0c;都可能希望通过增加硬盘来扩展存储容量。然而&#xff0c;一台电脑究竟最多能接多少个硬盘&#xff1…

【玩转全栈】---- Django 基于 Websocket 实现群聊(解决channel连接不了)

学习视频&#xff1a; 14-11 群聊&#xff08;一&#xff09;_哔哩哔哩_bilibili 目录 Websocket 连接不了&#xff1f; 收发数据 断开连接 完整代码 聊天室的实现 聊天室一 聊天室二 settings 配置 consumer 配置 多聊天室 Websocket 连接不了&#xff1f; 基于这篇博客&…

如何快速解决django报错:cx_Oracle.DatabaseError: ORA-00942: table or view does not exist

我们在使用django连接oracle进行编程时&#xff0c;使用model进行表映射对接oracle数据时&#xff0c;默认表名组成结构为&#xff1a;应用名_类名&#xff08;如&#xff1a;OracleModel_test&#xff09;&#xff0c;故即使我们库中存在表test&#xff0c;运行查询时候&#…

本地安装git

下载git 通过官网 下载 &#xff1a;Git - Downloading Package 若此页面无法直达&#xff0c;请删掉download/win尝试 2.双击运行安装 选择安装目录&#xff1a; 选择配置&#xff0c;默认不动 git安装目录名 默认即可 Git 的默认编辑器&#xff0c;建议使用默认的 Vim 编辑器…

小程序内表格合并功能实现—行合并

功能介绍&#xff1a;支付宝小程序手写表格实现行内合并&#xff0c;依据动态数据自动计算每次需求合并的值&#xff0c;本次记录行内合并&#xff0c;如果列内合并&#xff0c;同理即可实现 前端技术&#xff1a;grid布局 display&#xff1a;grid 先看实现效果: axml&…

SSE协议介绍和python实现

概述&#xff1a; SSE&#xff08;Server-Sent Events&#xff09;协议是一种允许服务器向客户端实时推送更新的技术&#xff0c;基于HTTP协议&#xff0c;常用于实时数据推送特点&#xff1a; 单向通信&#xff1a;服务器向客户端推送数据&#xff0c;客户端无法发送数据。基…

甘肃旅游服务平台+论文源码视频演示

4 系统设计 4.1系统概要设计 甘肃旅游服务平台并没有使用C/S结构&#xff0c;而是基于网络浏览器的方式去访问服务器&#xff0c;进而获取需要的数据信息&#xff0c;这种依靠浏览器进行数据访问的模式就是现在用得比较广泛的适用于广域网并且没有网速限制要求的小程序结构&am…

WebRTC中音视频服务质量QoS之FEC+NACK调用流程

WebRTC中音视频服务质量QoS之FECNACK调用流程 WebRTC中音视频服务质量QoS之FECNACK调用流程 WebRTC中音视频服务质量QoS之FECNACK调用流程前言一、WebRTC中FEC基础原理1. FEC基础操作 异或操作XOR2、 FEC中 行向和纵向 计算3、 WebRTC中 媒体包分组和生成FEC的包数① kFecRateT…

神经网络知识点整理

目录 ​一、深度学习基础与流程 二、神经网络基础组件 三、卷积神经网络&#xff08;CNN&#xff09;​编辑 四、循环神经网络&#xff08;RNN&#xff09;与LSTM 五、优化技巧与调参 六、应用场景与前沿​编辑 七、总结与展望​编辑 一、深度学习基础与流程 机器学习流…

远程办公新体验:用触屏手机流畅操作电脑桌面

在数字化浪潮的推动下&#xff0c;远程办公已从“应急选项”转变为职场常态。无论是居家隔离、差旅途中&#xff0c;还是咖啡厅临时办公&#xff0c;高效连接公司电脑的需求从未如此迫切。然而&#xff0c;传统的远程控制软件常因操作复杂、画面卡顿或功能限制而影响效率。如今…

【面试八股】:常见的锁策略

常见的锁策略 synchronized &#xff08;标准库的锁不够你用了&#xff09;锁策略和 Java 不强相关&#xff0c;其他语言涉及到锁&#xff0c;也有这样的锁策略。 1. 悲观锁&#xff0c;乐观锁&#xff08;描述的加锁时遇到的场景&#xff09; 悲观锁&#xff1a;预测接下来…

【python】OpenCV—Hand Detection

文章目录 1、功能描述2、代码实现3、效果展示4、完整代码5、参考6、其它手部检测和手势识别的方案 更多有趣的代码示例&#xff0c;可参考【Programming】 1、功能描述 基于 opencv-python 和 mediapipe 进行手部检测 2、代码实现 导入必要的库函数 import cv2 import media…

Flink中聚合算子介绍

前言 在flink api中&#xff0c;聚合算子是非常常用的。所谓的聚合就是在分组的基础上做比较计算的操作。下面通过几个简单案例来说明聚合算子的用法和注意事项。 聚合算子案例 因为flink的api操作流程比较固定&#xff0c;从获取执行环境》获取数据源》执行数据转换操作》输…

【基础】Windows 中通过 VSCode 使用 GCC 编译调试 C++

准备 安装 VSCode 及 C 插件。通过 MSYS2 安装 MinGW-w64 工具链&#xff0c;为您提供必要的工具来编译代码、调试代码并配置它以使用IntelliSense。参考&#xff1a;Windows 中的 Linux 开发工具链 验证安装&#xff1a; gcc --version g --version gdb --version三个核心配…