异步、回调函数和解决回调函数的方法

1. 同步和异步

1.1 同步

概念:按照代码的顺序执行任务

 console.log(1);
 console.log(2);

先输出1,在输出2
同步的缺陷
当遇到一个比较耗时间的任务,只能等该任务完成,后面的任务才能执行,后面的任务需要等待很长时间

1.2 异步

概念:

JS中的异步,需要异步语句:setInterval、setTimeout、Ajax、Node.js……等等
如果有异步语句了,那么一定是异步的。如果没有异步语句,那就不是异步的。

 console.log(1);
 setInterval(function(){
  console.log('❤')
 },1000)
 console.log(2);
image.png
 console.log(1);
 setInterval(function(){
  console.log('❤')
 },0)
 console.log(2);
image.png

从上往下:碰到定时函数和事件函数到一边去排队

2 回调函数

2.1 概念

是什么

是函数的应用方式,出现在两个函数之间
异步的语句做完之后要做的事情

什么形式

  • 把函数a当做参数传递到函数b中
  • 在函数b中以形参的方式进行调用
function a(cb){
  cb()
}
function b(){
  console.log('函数b')
}
a(b)
image.png

为什么需要

因为执行异步行为完毕之后需要做一些事情,由于无法预知异步行为什么时候完成

3. 回调地狱

出现回调地狱的原因

  • 一个回调函数嵌套一个回调函数,就会出现一个嵌套结构。
  • 嵌套的多了就会出现回调地狱

导致的结果

没有可维护性和可读性的代码

ajax({
  url: '我是第一个请求',
  success (res) {
    // 现在发送第二个请求
    ajax({
      url: '我是第二个请求',
      data: { a: res.a, b: res.b },
      success (res2) {
        // 进行第三个请求
        ajax({
          url: '我是第三个请求',
          data: { a: res2.a, b: res2.b },
                success (res3) { 
            console.log(res3) 
          }
        })
      }
    })
  }
})

回调地狱,其实就是回调函数嵌套过多导致的

  • 当代码成为这个结构以后,已经没有维护的可能了

4. 解决回调函数的方法

  • promise

  • async/await

(1)promise

是什么

  • 解决回调地狱的方案之一
  • 把回调地狱写的优雅的方案之一

目的

  • 解决回调地狱的问题
  • 使代码拥有可读性和可维护性

做什么

  • 异步的事情
  • 成功的时候给成功的回调
  • 失败的时候给失败的回调

三种状态两个结果

继续==> pending==> 正在进行时
成功==> fulfilled==> 过去完成时
失败==> rejected==> 过去完成时
pending 继续,正在做异步的事情
fulfilled 成功,结果 (resolved)
rejected 失败,结果

类比在ajax请求里面

=> pending 就是正在请求(网络传输过程中)
=> fulfilled 就是请求成功了(网络环境允许请求, 可以连接到服务器拿到结果)
=> rejected 就是请求失败了(突然没有网了, 掉线了, 不会再有结果回来了)

怎么用

Promise的基本语法

let p = new Promise(function(resolve,reject){
    //在这里做一个异步的事情
   // resolve() ==> 成功的回调,调用then里面的函数
   // reject() ==>失败的回调,调用catch里面的函数
})
p.then(function(){})
p.catch(function(){})
  1. 遇到了一个 Promise帮我发送一个 ajax 请求
    -> 发现这个是一个异步的 ajax
    -> 就把这个 异步的 ajax 往后放
    2. 遇到了一个 p1.then(function a() {})
    -> 就把这个回调函数帮我们准备好了传递到 promise 里面
    3. 遇到了一个 p1.catch(function b() {})
    -> 就把这个回调函数帮我们准备好了传递到 promise 里面
    到这个时候
    -> 所有的同步代码都执行完毕了
    -> a 和 b 函数只是传递到了 promise 里面, 但是没有调用
    -> 接下来就该异步的 ajax 执行了
    4. 等到 ajax 结束的时候
    -> 根据 ajax 的状态或者说成功还是失败
    -> 来调用之前就准备好的成功的回调或者失败的回调

promise链式调用

  • then 是成功的回调
  • 只要then里面再次return一个promise对象
  • 在这个 then 后面继续 then
 var p1 = new Promise(function (resolve, reject) {
      // 异步的事情先用定时器模拟
      setTimeout(function () {
        resolve() // 成功的回调
      }, 1000)
    })

    // 1s 以后执行这个 then 函数
    p1.then(function () {
      console.log('我是 p1 的 then 函数里面的代码')
      // 准备第二个 promise 对象
      var p2 = new Promise(function (resolve, reject) {
        setTimeout(function () {
          resolve() // 调用的第二个 then
        }, 1000)
      })

      // p2 是一个 promise 对象
      // 返回一个 promise 对象
      return p2
    }).then(function () {
      // 我会在 p2 的 resolve() 的时候执行
      console.log('我是 p1 then 后面的第二个 then')
    })

    // 在 p1 的 then 里面 return 了一个新的 promise 对象

(2) async / await

是什么

回调地狱的终极解决方案

  • es7 的语法
  • 用async/await语法把异步代码写的看起来像一个同步的代码

本质还是异步代码

怎么用

  • async关键字写在函数的前面,就把该函数变成了一个异步函数
  • await 是一个写在 异步函数 里面的关键字
  • await 关键字后面必须是一个promise对象

有了以上三个条件,就可以把本该在promise的then回调里面的接受的结果,放在await关键字前面定义个变量来接受

promise 对象
有了 await 以后
把本该在 then 里面接受的结果
在 await 前面定义一个变量来接受

  function pGetSend(url){
  var p1 = new Promise(function(resolve,reject){
      var xhr = new XMLHttpRequest();
      xhr.open('GET',url);
      xhr.onload = function(){
          resolve(xhr.responseText)
    }
    xhr.send()
 })
  return p1;
 }
// 使用 async
  async function fn(){
    console.log('开始1')
    var res = await pGetSend('./server/a.php')
    var result = JSON.parse(res)
    console.log(result);
    console.log('结束1')
 }
// 不使用 async
 function fn1(){
  console.log('开始2');
    var res = pGetSend('./server/a.php').then(function(res){
    console.log('第二次:'+res)
 })
   console.log('结束2');
 }

 fn()
 fn2()

运行结果:

image.png

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