JS多异步操作的并行与串行(异步循环)

多异步操作的并行与串行

前言

日常开放过程中经常遇到如下场景:

  • 多张图片同时上传,在所有图片都上传完毕,拿到所有url,再提交表单数据(异步并行)
  • 多个连续的请求,并且下一个请求需要使用到上一个请求的数据(异步串行)

遇到这样的场景是不是迷茫,怎么去控制串行与并行喃!!!

异步串行(多个异步操作之间有关联)

实战场景:
  • 需要按照顺序,调用多个接口。清空用户信息,需要先清除用户的一些信息,在亲空账号
  • 计算某些数据,需要执行多个请求。比如工资,需要获取考勤、绩效,上下接口之间有联系
题目场景:

我有一个数组,需要依次处理一个数组,处理的过程是一个异步操作。

解决办法:
  1. 回调地狱:顾名思义,就是采用回调函数来解决。这也是老前端代码采用最多的方式

    setTimeout(function () {
        console.log(arr[0])    //1
        setTimeout(function () {
            console.log(arr[1])  //2
            setTimeout(function () { 
                console.log(arr[2])  //3
                console.log("-----------所有异步执行完毕-----------")
            }, 1000)
        }, 1000)
    }, 1000)
    
  2. Promise:ES6提出的解决回调地狱的方案,从回调变成链式,但是更为繁琐

    new Promise(function (reslove, reject) {
        setTimeout(function () {
            console.log(arr[0])
            reslove()
        }, 1000);
    }).then(function () {
        return new Promise(function (reslove, reject) {
            setTimeout(function () {
                console.log(arr[1])  //2
                reslove()
            }, 1000);
        })
    }).then(function () {
        return new Promise(function (reslove, reject) {
            setTimeout(function () {
                console.log(arr[2])   //3
                console.log("-----------所有异步执行完毕-----------")
                reslove()
            }, 1000);
        })
    })
    
  3. 递归函数:把异步封装为一个函数,本质和回调函数差不多,但是不会出现回调地狱

let i = 0
function asyncFun() {
    setTimeout(function () {
        console.log(arr[i])
        i++
        if (i < arr.length) {
            asyncFun()
        } else {
            console.log("-----------所有异步执行完毕-----------")
        }
    }, 1000)
}
asyncFun()
  1. asyn/await(常用):个人感觉这是最简单的版本,也是最舒服的

     async function asyncFun() {
         for (let i = 0; i < arr.length; i++) {
             await new Promise(function (reslove, reject) {
                 setTimeout(function () {
                     console.log(arr[i])
                     reslove()
                 }, 1000);
             })
         }
         console.log("-----------所有异步执行完毕-----------")
     }
    asyncFun()
    

异步并行(多个异步操作之间有无关联)

实战场景:
  • 微信小程序上传图片,因为API限制,一次只能上传一张。
    • 异步串行: 一次上传一张图片,等上张图片上传完毕,在上传下一张(不推荐,因为这样上传完毕的事件会很慢,如果图片很多,用户体验很不好)
    • 异步并行:一起上传,会减少很多时间
题目场景:

我有三个文件,把目录路径放在数组,去异步读取所有文件,在读取完毕后再进行其他操作。

解决办法:
  1. 循环配合索引:设置一个变量,记录异步成功的数量,再在每次异步成功的回调函数里面判断是否所有文件都成功了

    let files = ["a.txt", "b.txt", "c.txt"]
    let rNmu=0 //成功读取文件的数量
    for (let i = 0; i < files.length; i++) {
        setTimeout(function () {
            console.log(files[i] + "已读取")
            rNmu++
            if (rNmu >= files.length) {   //判断是否所有文件都读取完毕
                console.log("-----------文件读取完毕-----------")
            }
        }, 1000);
    }
    
  2. map配合Promise.all(常用):通过map生成 Promise列表,在调用 Promise.all函数

// 方案1
let proList = files.map(function (name) {
    return new Promise(function (reslove, reject) {
        setTimeout(function () {
            console.log(name + "已读取")
            reslove()
        }, 1000);
    })
})
Promise.all(proList).then(function () {
    console.log("-----------文件读取完毕-----------")
})


//方案二,配合async、await使用
let proList = files.map(async function (name) {
    await new Promise(function (reslove, reject) {
        setTimeout(function () {
            console.log(name + "已读取")
            reslove()
        }, 1000);
    })
})
Promise.all(proList).then(function () {
    console.log("-----------文件读取完毕-----------")
})
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 219,110评论 6 508
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,443评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 165,474评论 0 356
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,881评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,902评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,698评论 1 305
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,418评论 3 419
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,332评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,796评论 1 316
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,968评论 3 337
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,110评论 1 351
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,792评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,455评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,003评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,130评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,348评论 3 373
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,047评论 2 355

推荐阅读更多精彩内容