Javascript 异步编程方法汇总

  1. 回调函数
    假定有两个函数f1和f2,后者等待前者的执行结果。
    f1(); f2();
    如果f1是一个很耗时的任务,可以考虑改写为f1,把f2写为f1的回调函数。
    function f1(callback){
       setTimeout(function(){
            //f1代码
            callback();
        },1000);
    }
    
    执行的时候就变成了f1(f2); 采用这种方法把同步变成异步,f1不会堵塞程序运行,相当于先执行程序的主要逻辑,将耗时都部分推迟执行。
    回调函数优点:简单、容易理解和部署;缺点:不利于代码的阅读和维护。
    一个同步(阻塞)中使用回调的例子,目的是在func1代码执行完成后执行func2代码。
    var func1 = function(callback){
           // do something 
          (callback && typeof(callback) === "function") && callback();
    }
    func1(func2);
    var func2 = function(){}
    

更多回调函数请参考:

  1. 事件监听
    任务的执行不取决于代码的顺序,而取决于某件事件是否发生。
    为f1绑定一个事件:f1.on('done', f2); 当f1发生done事件,就执行f2。然后对f1进行改写为
    function f1(){
    setTimeout(function*(){
         // f1的任务代码
         f1.trigger('done');
        },1000);
    }
    
    优点:比较容易理解,绑定多个事件,每个事件可以指定多个回调函数,去耦合,实现模块化。
  2. 发布/订阅
    例如:jquery中一个插件:Tiny Pub/Sub
    1. f2向信号中心"jquery订阅"done信号: jquery.subscribe("done", f2);
    2. f1函数如下:
      function f1(){
          setTimeout(function(){
                // f1的执行代码
                 jquey.publish('done');
          }, 1000);
       }
      
      jquey.publish('done');就是f1执行完毕之后,向信号中心发送done信号,从而引发f2的执行。f2执行后,也可以取消订阅jquery.unsubscribe("done", f2);
  3. Promises对象:为异步编程提供统一接口。
    思想:每一个异步任务返回一个Promise对象,该对象有一个then方法,允许指向回调函数。
    Promise的三种状态:
    • Pending:Promise对象实例创建时候的初始状态。
    • Fulfilled:成功的状态
    • Rejected:失败的状态
      Promise ---> resolved ---> then(回调callback) 或者 ---> rejected ---> catch(回调callback)。Promise一旦从等待状态变成其他状态就永远不能改变状态了。
       const instance = new Promise((resolve, reject) => {
         // 一些异步操作
         if(/*异步操作成功*/) {
             resolve(value);
         } else {
             reject(error);
             }
         }
       })
       instance.then(value => {
           // do something...
        }, error => {
             // do something...
        })
      
      构造函数内部的代码立刻执行。
      then 方法会返回一个新的 Promise 实例,可以分两种情况来看:
      • 指定返回值是新的 Promise 对象,如return new Promise(...),这种情况没啥好说的,由于返回的是 Promise,后面显然可以继续调用then方法。
      • 返回值不是Promise, 如:return 1 这种情况还是会返回一个 Promise,并且这个Promise 立即执行回调 resolve(1)。所以仍然可以链式调用then方法。(注:如果没有指定return语句,相当于返回了undefined)。
      1. 示例一
      function sayHi(name) {
         return new Promise((resolve, reject) => {
               setTimeout(() => {
                   resolve(name);
                }, 2000)
          })
      }
      sayHi('张三')
         .then(name => {
               console.log(`你好, ${name}`);
               return sayHi('李四');    // 最终 resolved 函数中的参数将作为值传递给下一个then
            })
           // name 是上一个then传递出来的参数
         .then(name => {                
             console.log(`你好, ${name}`);
             return sayHi('王二麻子');
           })
         .then(name => {
             console.log(`你好, ${name}`);
         })
       // 你好, 张三
       // 你好, 李四
       // 你好, 王二麻子
      
      1. 示例二
       read('./file-01.txt', 'utf8')
          .then(data => {
            // 因为 read 方法会返回一个 promise ,返回read执行相当于返回一个 promise
            // 会将这个 promise 的执行成功的结果传递给下一次 then 的 resolve 
                return read(data, 'utf8');
            }, err => {
                console.log(err);
          })
          .then(data => {
            // 如果返回一个普通的值 ,会将这个普通值传递倒下一次 then 的成功的参数
              return [data];
           }, err => {
             console.log(err);
           })
          .then(data => {
            // 返回的是一个普通值
             console.log(data);
             // 没有写 return ,相当于返回一个 undefined ,下一个then的成功值则为 undefined
              }, err => {
              console.log(err);
            })
          .then(data => {
          // 上一个 then 没有返回值,默认值为 undefined
          // undefined 也算成功
            console.log(data);
            // 抛出错误,将传给下一个 then 的 reject 
              throw new Error('xxx');
          }, err => {
              console.log(err);
        })
         .then(null, err => {
            // 如果上一个 then 抛出错误,最近的 reject 会执行
            // reject 执行后默认下一个 then 会接收 undefined 
          console.log(err);
        })
        .then(data => {
            // 上一个 then 中失败没有返回值,默认为 undefined 
            console.log('resolve');        
          }, err => {
            console.log('reject');
        });
      
      1. 示例三:使用catch代替err
       read('./file-01.txt', 'utf8')
          .then(data => {
              return read(data, 'utf8');
          })
        .then(data => {
            return [data];
        })
        .then(data => {
            console.log(data);
        })
        .then(data => {
          console.log(data);
            // 抛出错误后,找到最近的接收错误方法
            // 如果所有的 then 都没有 reject 方法,则找最后一个 catch
            throw new Error('xxx');
          })
        .then(null)
        .then(data => {
            console.log('resolve');        
          })
        .catch(err => {
            console.log(err);
        });
      
  4. async/await
    • async/await是基于Promise实现的,不能用于普通的回调函数。
    • 和Promise一样,是非阻塞的。
    • 使异步代码看起来像是同步代码。
    • 一个函数如果加上async,那么该函数就会返回一个Promise,(如果指定的返回值不是Promise对象,也返回一个Promise,只不过立即 resolve ,处理方式同 then 方法,因此 async 函数通过 return 返回的值,会成为 then 方法中回调函数的参数。单独一个 async 函数,其实与Promise执行的功能是一样的。
    • await 就是异步等待,它等待的是一个Promise,因此 await 后面应该写一个Promise对象,如果不是Promise对象,那么会被转成一个立即 resolve 的Promise
    1. 实例一
        let fs = require('fs')
        function read(file) {
        return new Promise(function(resolve, reject) {
            fs.readFile(file, 'utf8', function(err, data) {
            if (err) reject(err)
            resolve(data)
            })
        })
        }
        async function readResult(params) {
        try {
            let p1 = await read(params, 'utf8')
            //await后面跟的是一个Promise实例,
            //await返回的结果直接return的结果,不需要进行新的then操作。
            let p2 = await read(p1, 'utf8')
            let p3 = await read(p2, 'utf8')
            console.log('p1', p1)
            console.log('p2', p2)
            console.log('p3', p3)
            return p3
        } catch (error) {
            console.log(error)
        }
        }
        readResult('1.txt').then( // async函数返回的也是个promise
        data => {
            console.log(data)
        },
        err => console.log(err)
        )
        // p1 2.txt
        // p2 3.txt
        // p3 结束
        // 结束
      
    2. 示例二
    function readAll() {
       read1()
       read2()//这个函数同步执行
      }
    async function read1() {
       let r = await read('1.txt','utf8')
       console.log(r)
     }
    async function read2() {
       let r = await read('2.txt','utf8')
       console.log(r)
     }
    readAll() // 2.txt 3.txt
    
    1. 示例三
    async function func() {
        try {
            const num1 = await 200;
            console.log(`num1 is ${num1}`);
            const num2 = await Promise.reject('num2 is wrong!');
            console.log(`num2 is ${num2}`);
            const num3 = await num2 + 100;
            console.log(`num3 is ${num3}`);
        } catch (error) {
            console.log(error);
        }
     }
    
    func();
    // num1 is 200
    // 出错了
    // num2 is wrong!
    

参考:
- Javascript异步编程的4种方法
- 异步方法的发展流程
- JS 异步编程六种方案

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

推荐阅读更多精彩内容

  • Promise 对象 Promise 的含义 Promise 是异步编程的一种解决方案,比传统的解决方案——回调函...
    neromous阅读 8,705评论 1 56
  • 弄懂js异步 讲异步之前,我们必须掌握一个基础知识-event-loop。 我们知道JavaScript的一大特点...
    DCbryant阅读 2,710评论 0 5
  • 前言 本文旨在简单讲解一下javascript中的Promise对象的概念,特性与简单的使用方法。并在文末会附上一...
    _暮雨清秋_阅读 2,194评论 0 3
  • 在ES6当中添加了很多新的API其中很值得一提的当然少不了Promise,因为Promise的出现,很轻松的就给开...
    嘿_那个谁阅读 3,668评论 2 3
  • 原文连接:https://blog.csdn.net/sinat_17775997/article/details...
    小豆soybean阅读 4,248评论 0 7