AJAX进阶(重点)

目录

[◆ 同步代码和异步代码](#◆ 同步代码和异步代码)

[◆ 回调函数地狱和 Promise 链式调用](#◆ 回调函数地狱和 Promise 链式调用)

什么是回调函数地狱?

[Promise - 链式调用](#Promise - 链式调用)

什么是Promise链式调用?

[Promise 链式应用 (重点)](#Promise 链式应用 (重点))

[◆ async 和 await 使用](#◆ async 和 await 使用)

async函数和await_捕获错误

[◆ 事件循环-EventLoop(重点)](#◆ 事件循环-EventLoop(重点))

为什么要学习事件循环?

什么是事件循环?

[事件循环 - 执行过程(重点)](#事件循环 - 执行过程(重点))

[事件循环 - 练习](#事件循环 - 练习)

[宏任务与微任务 (重点)](#宏任务与微任务 (重点))

[事件循环 - 经典面试题](#事件循环 - 经典面试题)

[◆ Promise.all 静态方法](#◆ Promise.all 静态方法)


◆ 同步代码和异步代码

什么是同步代码,什么是异步代码?

同步代码:

异步代码:

同步代码:逐行执行,需原地等待结果后,才继续向下执行

异步代码:调用后耗时,不阻塞代码继续执行(不必原地等待),在将来完成后触发一个回调函数

例子:回答打印数字的顺序是什么?

打印结果:1,4,2 点击按钮一次就打印一次 3

异步代码接收结果:使用回调函数

◆ 回调函数地狱和 Promise 链式调用

什么是回调函数地狱?

需求:展示默认第一个省,第一个城市,第一个地区在下拉菜单中

概念:在回调函数中嵌套回调函数,一直嵌套下去就形成了回调函数地狱

缺点:可读性差,异常无法捕获,耦合性严重,牵一发动全身

总结:简单讲。回调函数地狱就是在回调函数中嵌套回调函数,进而导致代码耦合度高,异常难以捕获

Promise - 链式调用

什么是Promise链式调用?

概念:依靠then() 方法会返回一个新生成的 Promise 对象特性,继续串联下一环任务,直到结束

细节:then() 回调函数中的返回值 ,会影响新生成的Promise 对象最终状态和结果

好处:通过链式调用,解决回调函数嵌套问题

Promise 链式应用 (重点)

目标:使用 Promise 链式调用,解决回调函数地狱问题

做法:每个 Promise 对象中管理一个异步任务,用 then 返回 Promise 对象,串联起来

html 复制代码
<!DOCTYPE html>
<html lang="en">

<head>
  <meta charset="UTF-8">
  <meta http-equiv="X-UA-Compatible" content="IE=edge">
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
  <title>Promise链式调用_解决回调地狱</title>
</head>

<body>
  <form>
    <span>省份:</span>
    <select>
      <option class="province"></option>
    </select>
    <span>城市:</span>
    <select>
      <option class="city"></option>
    </select>
    <span>地区:</span>
    <select>
      <option class="area"></option>
    </select>
  </form>
  <script src="https://cdn.jsdelivr.net/npm/axios/dist/axios.min.js"></script>
  <script>
    /**
     * 目标:把回调函数嵌套代码,改成Promise链式调用结构
     * 需求:获取默认第一个省,第一个市,第一个地区并展示在下拉菜单中
    */
    let pname = ''
    // 1. 得到-获取省份Promise对象
    axios({url: 'http://hmajax.itheima.net/api/province'}).then(result => {
      pname = result.data.list[0]
      document.querySelector('.province').innerHTML = pname
      // 2. 得到-获取城市Promise对象
      return axios({url: 'http://hmajax.itheima.net/api/city', params: { pname }})
    }).then(result => {
      const cname = result.data.list[0]
      document.querySelector('.city').innerHTML = cname
      // 3. 得到-获取地区Promise对象
      return axios({url: 'http://hmajax.itheima.net/api/area', params: { pname, cname }})
    }).then(result => {
      console.log(result)
      const areaName = result.data.list[0]
      document.querySelector('.area').innerHTML = areaName
    })
  </script>
</body>

</html>

总结:

  1. 什么是 Promise 的链式调用?
  • ➢ 使用 then 方法返回新 Promise 对象特性,一直串联下去
  1. then 回调函数中,return 的值会传给哪里?
  • ➢ 传给 then 方法生成的新 Promise 对象
  1. Promise 链式调用有什么用?
  • ➢ 解决回调函数嵌套问题

◆ async 和 await 使用

定义:

概念: 在 async 函数内,使用 await 关键字取代 then 函数,等待获取 Promise 对象成功状态的结果值

示例:

简单来说,当函数使用了async关键字,可以使用await关键字讲函数内的异步函数转化成同步函数

async函数和await_捕获错误

◆ 事件循环-EventLoop(重点)

为什么要学习事件循环?

掌握 JavaScript 是如何安排和运行代码的

什么是事件循环?

概念:

原因:JavaScript 单线程(某一刻只能执行一行代码),为了让耗时代码不阻塞其他代码运行,设计了事件循环模型

事件循环 - 执行过程(重点)

定义:执行代码和收集异步任务的模型,在调用栈空闲时,反复调用任务队列里回调函数的执行机制,就叫事件循环

总结:

  1. 什么是事件循环?
  • ➢ 执行代码和收集异步任务,在调用栈空闲时,反复调用任务队列里

回调函数执行机制

  1. 为什么有事件循环?
  • ➢ JavaScript 是单线程的,为了不阻塞 JS 引擎,设计执行代码的模型
  1. JavaScript 内代码如何执行?
  • ➢ 执行同步代码,遇到异步代码交给宿主浏览器环境执行
  • ➢ 异步有了结果后,把回调函数放入任务队列排队
  • ➢ 当调用栈空闲后,反复调用任务队列里的回调函数

事件循环 - 练习

使用模型,分析代码执行过程

宏任务与微任务 (重点)

ES6 之后引入了 Promise 对象, 让 JS 引擎也可以发起异步任务

异步任务分为:

宏任务:由浏览器环境执行的异步代码

例子:

微任务:由 JS 引擎环境执行的异步代码

例子:

Promise 本身是同步的,而then和catch回调函数是异步的

使用图解-分析代码执行顺序

JS代码的执行流程:script代码块-》代码块里的同步任务-》微任务-》宏任务

事件循环 - 经典面试题

答案:1756234

总结:

  1. 什么是宏任务?
  • ➢ 浏览器执行的异步代码
  • ➢ 例如:JS 执行脚本事件,setTimeout/setInterval,AJAX请求完成事件,用户交互事件
  1. 什么是微任务?
  • ➢ JS 引擎执行的异步代码
  • ➢ 例如:Promise对象.then()的回调
  1. JavaScript 内代码如何执行?
  • ➢ 执行第一个 script 脚本事件宏任务,里面同步代码
  • ➢ 遇到 宏任务/微任务 交给宿主环境,有结果回调函数进入对应队列
  • ➢ 当执行栈空闲时,清空微任务队列,再执行下一个宏任务,从1再来

◆ Promise.all 静态方法

概念:合并多个 Promise 对象,等待所有同时成功完成(或某一个失败),做后续逻辑

语法:

案例:

需求:同时请求"北京","上海","广州","深圳"的天气并在网页尽可能同时显示

html 复制代码
<!DOCTYPE html>
<html lang="en">

<head>
  <meta charset="UTF-8">
  <meta http-equiv="X-UA-Compatible" content="IE=edge">
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
  <title>Promise的all方法</title>
</head>

<body>
  <ul class="my-ul"></ul>
  <script src="https://cdn.jsdelivr.net/npm/axios/dist/axios.min.js"></script>
  <script>
    /**
     * 目标:掌握Promise的all方法作用,和使用场景
     * 业务:当我需要同一时间显示多个请求的结果时,就要把多请求合并
     * 例如:默认显示"北京", "上海", "广州", "深圳"的天气在首页查看
     * code:
     * 北京-110100
     * 上海-310100
     * 广州-440100
     * 深圳-440300
    */
    // 1. 请求城市天气,得到Promise对象
    const bjPromise = axios({ url: 'http://hmajax.itheima.net/api/weather', params: { city: '110100' } })
    const shPromise = axios({ url: 'http://hmajax.itheima.net/api/weather', params: { city: '310100' } })
    const gzPromise = axios({ url: 'http://hmajax.itheima.net/api/weather', params: { city: '440100' } })
    const szPromise = axios({ url: 'http://hmajax.itheima.net/api/weather', params: { city: '440300' } })

    // 2. 使用Promise.all,合并多个Promise对象
    const p = Promise.all([bjPromise, shPromise, gzPromise, szPromise])
    p.then(result => {
      // 注意:结果数组顺序和合并时顺序是一致
      console.log(result)
      const htmlStr = result.map(item => {
        return `<li>${item.data.data.area} --- ${item.data.data.weather}</li>`
      }).join('')
      document.querySelector('.my-ul').innerHTML = htmlStr
    }).catch(error => {
      console.dir(error)
    })
  </script>
</body>

</html>
相关推荐
以对_6 分钟前
uview表单校验不生效问题
前端·uni-app
Zheng1131 小时前
【可视化大屏】将柱状图引入到html页面中
javascript·ajax·html
程序猿小D1 小时前
第二百六十七节 JPA教程 - JPA查询AND条件示例
java·开发语言·前端·数据库·windows·python·jpa
john_hjy1 小时前
【无标题】
javascript
奔跑吧邓邓子2 小时前
npm包管理深度探索:从基础到进阶全面教程!
前端·npm·node.js
软件开发技术深度爱好者2 小时前
用HTML5+CSS+JavaScript庆祝国庆
javascript·css·html5
前端李易安2 小时前
ajax的原理,使用场景以及如何实现
前端·ajax·okhttp
汪子熙2 小时前
Angular 服务器端应用 ng-state tag 的作用介绍
前端·javascript·angular.js
Envyᥫᩣ3 小时前
《ASP.NET Web Forms 实现视频点赞功能的完整示例》
前端·asp.net·音视频·视频点赞
Мартин.6 小时前
[Meachines] [Easy] Sea WonderCMS-XSS-RCE+System Monitor 命令注入
前端·xss