AJAX笔记进阶篇

黑马程序员视频地址:

AJAX-Day04-01.同步代码和异步代码https://www.bilibili.com/video/BV1MN411y7pw?vd_source=0a2d366696f87e241adc64419bf12cab&spm_id_from=333.788.videopod.episodes&p=47https://www.bilibili.com/video/BV1MN411y7pw?vd_source=0a2d366696f87e241adc64419bf12cab&spm_id_from=333.788.videopod.episodes&p=47

同步代码和异步代码


回调函数地狱与解决方法

回调函数地狱

需求:展示默认第一个省,第一个城市,第一个地区在下拉菜单中
概念:在回调函数中嵌套回调函数,一直嵌套下去就形成了回调函数地狱
缺点:可读性差,异常无法通过一次代码捕获,耦合性严重,牵一发动全身

/*** 目标:演示回调函数地狱* 需求:获取默认第一个省,第一个市,第一个地区并展示在下拉菜单中* 概念:在回调函数中嵌套回调函数,一直嵌套下去就形成了回调函数地狱* 缺点:可读性差,异常无法写一次代码获取,耦合性严重,牵一发动全身*/// 1. 获取默认第一个省份的名字axios({url: 'http://hmajax.itheima.net/api/province'}).then(result => {const pname = result.data.list[0]document.querySelector('.province').innerHTML = pname// 2. 获取默认第一个城市的名字axios({url: 'http://hmajax.itheima.net/api/city', params: { pname }}).then(result => {const cname = result.data.list[0]document.querySelector('.city').innerHTML = cname// 3. 获取默认第一个地区的名字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})})}).catch(error => {console.dir(error) //无法通过一次catch捕获任意一个axios错误})

解决方法一:Promise 链式调用 

deepseek给出的解释:

Promise 链式调用:通过 Promise 状态(Pending/Fulfilled/Rejected)的不可变性,严格保证执行顺序。

Promise 的 then() 方法调用后会返回一个新的 Promise 对象。这是 Promise 链式调用的核心机制。以下是关键点解释:


1. 返回一个新的 Promise

  • 原 Promise 不受影响:调用 then() 不会修改原始的 Promise 对象,而是生成一个新的 Promise。

  • 链式调用的基础:正是通过返回新 Promise,才能继续调用下一个 .then() 或 .catch(),形成链式结构。

2. 新 Promise 的状态由回调决定

  • 回调返回值决定状态:新 Promise 的状态(fulfilled 或 rejected)和值,由 then() 中传入的回调函数决定:

    • 如果回调返回普通值(非 Promise),新 Promise 会以该值变为 fulfilled

    • 如果回调返回一个新的 Promise,新 Promise 会“跟随”这个返回的 Promise 的状态。

    • 如果回调抛出错误,新 Promise 会变为 rejected

博主重述:调用一次axios时,基于axios内部代码,会自动new一个promise对象,并且执行内部代码,成功会调用resolve()方法,此时原promise对象被标记为成功,然后会自动调用then方法,并且返回一个新对象,而新对象的标记与返回值有关,若返回值也为一个新promise对象,则会与其的标志保持一致,此时如果新对象执行成功(即调用resolve()方法),则会被标记为成功,从而执行then方法,如此循环,一级一级读取then方法,倘若某一级的promise对象被标记失败,则会找catch方法,即跳过其他then而直接到最后一行找catch

/*** 目标:把回调函数嵌套代码,改成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}).catch(error => {console.log(error)})//一次catch捕获即可获取任意某一次axios错误

解决方法二:async 和await 使用

async:标记函数为异步,使其支持 await。普通声明函数为同步函数
await:暂停异步函数,等待异步操作完成,让代码逻辑更清晰。

await 会暂停当前 async 函数的执行,但不会阻塞线程。其他代码(如事件循环)可以继续运行。 

    /*** 目标:掌握async和await语法,解决回调函数地狱* 概念:在async函数内,使用await关键字,获取Promise对象"成功状态"结果值* 注意:await必须用在async修饰的函数内(await会阻止"异步函数内"代码继续执行,原地等待结果)*/// 1. 定义async修饰函数async function getData() {// 2. await等待Promise对象成功的结果,同时会返回promise对象成功执行返回的值(即then里的result)const pObj = await axios({url: 'http://hmajax.itheima.net/api/province'})const pname = pObj.data.list[0]const cObj = await axios({url: 'http://hmajax.itheima.net/api/city', params: { pname }})const cname = cObj.data.list[0]const aObj = await axios({url: 'http://hmajax.itheima.net/api/area', params: { pname, cname }})const areaName = aObj.data.list[0]document.querySelector('.province').innerHTML = pnamedocument.querySelector('.city').innerHTML = cnamedocument.querySelector('.area').innerHTML = areaName}getData()

捕获错误:

    /*** 目标:async和await_错误捕获*/async function getData() {// 1. try包裹可能产生错误的代码try {const pObj = await axios({ url: 'http://hmajax.itheima.net/api/province' })const pname = pObj.data.list[0]const cObj = await axios({ url: 'http://hmajax.itheima.net/api/city', params: { pname } })const cname = cObj.data.list[0]const aObj = await axios({ url: 'http://hmajax.itheima.net/api/area', params: { pname, cname } })const areaName = aObj.data.list[0]document.querySelector('.province').innerHTML = pnamedocument.querySelector('.city').innerHTML = cnamedocument.querySelector('.area').innerHTML = areaName} catch (error) {// 2. 接着调用catch块,接收错误信息// 如果try里某行代码报错后,try中剩余的代码不会执行了console.dir(error)}}getData()

事件循环-EventLoop

1. 什么是事件循环?

➢ 执行代码和收集异步任务,在调用栈空闲时,反复调用任务队列里 回调函数执行机制

2. 为什么有事件循环?

➢ JavaScript 是单线程的,为了不阻塞JS 引擎,设计执行代码的模型

3. JavaScript 内代码如何执行?

➢ 执行同步代码,遇到异步代码交给宿主浏览器环境执行

➢ 异步有了结果后,把回调函数放入任务队列排队

➢ 当调用栈空闲后,反复调用任务队列里的回调函数 


宏任务与微任务


Promise.all静态方法

    // 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)}).catch(error => {console.log(error)})

 案例1—商品分类

目标:获取一级分类,并通过一级分类获取二级分类,并渲染到页面上

写法一:axios嵌套

   //1. 获取所有一级分类数据axios({url: "https://hmajax.itheima.net/api/category/top"}).then(result => {const topList = result.data.data//2. 遍历id,创建获取二级分类请求const subList = topList.map(item => {return axios({url: "https://hmajax.itheima.net/api/category/sub",params: {id: item.id}})})//3. 合并所有二级分类Promise对象const allList= Promise.all(subList)//4. 等待同时成功后,渲染页面allList.then(result => {const data = result.map(item => {return `<div class="item"><h3>${item.data.data.name}</h3><ul>${item.data.data.children.map(list => {return `<li><a href="javascript:;"><img src="${list.picture}"><p>${list.name}</p></a></li>`}).join("")}</ul></div>`}).join("")document.querySelector(".sub-list").innerHTML = data})})

 写法二:async

async function getData()
{try{//1. 获取所有一级分类数据const topList = await axios({url: "https://hmajax.itheima.net/api/category/top"})const topListData = topList.data.data//2. 遍历id,创建获取二级分类请求const subList = topListData.map(subItem => {return axios({url: "https://hmajax.itheima.net/api/category/sub", params: {id: subItem.id}})})//3. 合并所有二级分类Promise对象,并等待同时成功const subListData = await Promise.all(subList)//4.成功后渲染页面const data = subListData.map(subData => {return `<div class="item"><h3>${subData.data.data.name}</h3><ul>${subData.data.data.children.map(subItemData => {return `<li><a href="javascript:;"><img src="${subItemData.picture}"><p>${subItemData.name}</p></a></li>`}).join("")}</ul></div>`}).join("")document.querySelector(".sub-list").innerHTML = data}catch(error){//出错执行}}
getData()

案例2—学习反馈

目标一:选择省份、城市、地区数据 


//1.1 设置省份下拉菜单数据
axios({url: "https://hmajax.itheima.net/api/province"}).then(provinceResult => {const provinceList = provinceResult.data.list//映射数据const province = provinceList.map(province => {return `<option value="${province}">${province}</option>`}).join("")//渲染到页面中document.querySelector(".province").innerHTML = `<option value="">省份</option>` + province
})//获取相应城市
document.querySelector(".province").addEventListener("change", e => {//获取城市数据axios({url: "https://hmajax.itheima.net/api/city", params: {pname: e.target.value}}).then(cityResult => {const cityList = cityResult.data.listconst city = cityList.map(city => {return `<option value="${city}">${city}</option>`}).join("")document.querySelector(".city").innerHTML = `<option value="">城市</option>` + city})//清除地区数据document.querySelector(".area").innerHTML = `<option value="">地区</option>`
})//获取相应地区
document.querySelector(".city").addEventListener("change", e => {//获取数据axios({url: "https://hmajax.itheima.net/api/area", params: {pname:document.querySelector('.province').value ,cname:e.target.value}}).then(areaResult => {const areaList = areaResult.data.listconst area = areaList.map(area => {return `<option value="${area}">${area}</option>`}).join("")document.querySelector(".area").innerHTML = `<option value="">地区</option>` + area})
})

目标二:收集数据,提交数据

/*** 目标2:收集数据提交保存*  2.1 监听提交的点击事件*  2.2 依靠插件收集表单数据*  2.3 基于axios提交保存,显示结果*/
// 2.1 监听提交的点击事件
document.querySelector('.submit').addEventListener('click', async () => {// 2.2 依靠插件收集表单数据const form = document.querySelector('.info-form')const data = serialize(form, { hash: true, empty: true })console.log(data)// 2.3 基于axios提交保存,显示结果try {const result = await axios({url: 'http://hmajax.itheima.net/api/feedback',method: 'POST',data})console.log(result)alert(result.data.message)} catch (error) {console.dir(error)alert(error.response.data.message)}
})

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

利用Muduo库实现简单且健壮的Echo服务器

一、muduo网络库主要提供了两个类&#xff1a; TcpServer&#xff1a;用于编写服务器程序 TcpClient&#xff1a;用于编写客户端程序 二、三个重要的链接库&#xff1a; libmuduo_net、libmuduo_base、libpthread 三、muduo库底层就是epoll线程池&#xff0c;其好处是…

【学术投稿-2025年计算机视觉研究进展与应用国际学术会议 (ACVRA 2025)】从计算机基础到HTML开发:Web开发的第一步

会议官网&#xff1a;www.acvra.org 简介 2025年计算机视觉研究进展与应用&#xff08;ACVRA 2025&#xff09;将于2025年2月28-3月2日在中国广州召开&#xff0c;将汇聚世界各地的顶尖学者、研究人员和行业专家&#xff0c;聚焦计算机视觉领域的最新研究动态与应用成就。本次…

Rust 所有权特性详解

Rust 所有权特性详解 Rust 的所有权系统是其内存安全的核心机制之一。通过所有权规则&#xff0c;Rust 在编译时避免了常见的内存错误&#xff08;如空指针、数据竞争等&#xff09;。本文将从堆内存与栈内存、所有权规则、变量作用域、String 类型、内存分配、所有权移动、Cl…

【Git】一、初识Git Git基本操作详解

文章目录 学习目标Ⅰ. 初始 Git&#x1f4a5;注意事项 Ⅱ. Git 安装Linux-centos安装Git Ⅲ. Git基本操作一、创建git本地仓库 -- git init二、配置 Git -- git config三、认识工作区、暂存区、版本库① 工作区② 暂存区③ 版本库④ 三者的关系 四、添加、提交更改、查看提交日…

【Envi遥感图像处理】009:envi5.6设置中文界面的方法

ENVI软件从5.0版本开始,界面发生了大的变化,并开始支持中文。本文讲述envi5.6设置中文界面的方法。 文章目录 一、中文界面预览二、设置中文界面三、注意事项一、中文界面预览 以下为envi5.6新版的中文界面: 二、设置中文界面 打开英文版的envi5.6软件,首先需要从安装App…

014-STM32单片机实现矩阵薄膜键盘设计

1.功能说明 本设计主要是利用STM32驱动矩阵薄膜键盘&#xff0c;当按下按键后OLED显示屏上会对应显示当前的按键键值&#xff0c;可以将此设计扩展做成电子秤、超市收银机、计算器等需要多个按键操作的单片机应用。 2.硬件接线 模块管脚STM32单片机管脚矩阵键盘行1PA0矩阵键盘…

鸿蒙Harmony-双向数据绑定MVVM以及$$语法糖介绍

鸿蒙Harmony-双向数据绑定MVVM以及$$语法糖介绍 1.1 双向数据绑定概念 在鸿蒙&#xff08;HarmonyOS&#xff09;应用开发中&#xff0c;双向数据改变&#xff08;或双向数据绑定&#xff09;是一种让数据模型和UI组件之间保持同步的机制&#xff0c;当数据发生变化时&#x…

Chromium132 编译指南 - Android 篇(一):编译前准备

1. 引言 欢迎来到《Chromium 132 编译指南 - Android 篇》系列的第一部分。本系列指南将引导您逐步完成在 Android 平台上编译 Chromium 132 版本的全过程。Chromium 作为一款由 Google 主导开发的开源浏览器引擎&#xff0c;为众多现代浏览器提供了核心驱动力。而 Android 作…

通向AGI之路:人工通用智能的技术演进与人类未来

文章目录 引言:当机器开始思考一、AGI的本质定义与技术演进1.1 从专用到通用:智能形态的范式转移1.2 AGI发展路线图二、突破AGI的五大技术路径2.1 神经符号整合(Neuro-Symbolic AI)2.2 世界模型架构(World Models)2.3 具身认知理论(Embodied Cognition)三、AGI安全:价…

使用 DeepSeek-R1 与 AnythingLLM 搭建本地知识库

一、下载地址Download Ollama on macOS 官方网站&#xff1a;Ollama 官方模型库&#xff1a;library 二、模型库搜索 deepseek r1 deepseek-r1:1.5b 私有化部署deepseek&#xff0c;模型库搜索 deepseek r1 运行cmd复制命令&#xff1a;ollama run deepseek-r1:1.5b 私有化…

C++游戏开发实战:从引擎架构到物理碰撞

&#x1f4dd;个人主页&#x1f339;&#xff1a;一ge科研小菜鸡-CSDN博客 &#x1f339;&#x1f339;期待您的关注 &#x1f339;&#x1f339; 1. 引言 C 是游戏开发中最受欢迎的编程语言之一&#xff0c;因其高性能、低延迟和强大的底层控制能力&#xff0c;被广泛用于游戏…

计算机网络——三种交换技术

目录 电路交换——用于电话网络 电路交换的优点&#xff1a; 电路交换的缺点&#xff1a; 报文交换——用于电报网络 报文交换的优点&#xff1a; 报文交换的缺点&#xff1a; 分组交换——用于现代计算机网络 分组交换的优点&#xff1a; 分组交换的缺点 电路交换——…

35.Word:公积金管理中心文员小谢【37】

目录 Word1.docx ​ Word2.docx Word2.docx ​ 注意本套题还是与上一套存在不同之处 Word1.docx 布局样式的应用设计页眉页脚位置在水平/垂直方向上均相对于外边距居中排列&#xff1a;格式→大小对话框→位置→水平/垂直 按下表所列要求将原文中的手动纯文本编号分别替换…

小程序越来越智能化,作为设计师要如何进行创新设计

一、用户体验至上 &#xff08;一&#xff09;简洁高效的界面设计 小程序的特点之一是轻便快捷&#xff0c;用户期望能够在最短的时间内找到所需功能并完成操作。因此&#xff0c;设计师应致力于打造简洁高效的界面。避免过多的装饰元素和复杂的布局&#xff0c;采用清晰的导航…

全栈开发:使用.NET Core WebAPI构建前后端分离的核心技巧(二)

目录 配置系统集成 分层项目使用 筛选器的使用 中间件的使用 配置系统集成 在.net core WebAPI前后端分离开发中&#xff0c;配置系统的设计和集成是至关重要的一部分&#xff0c;尤其是在管理不同环境下的配置数据时&#xff0c;配置系统需要能够灵活、可扩展&#xff0c…

Linux——进程概念

目录 一、系统调用和库函数概念二、基本概念三、描述进程-PCB3.1 task_struct-PCB的一种3.2 task_ struct内容分类 四、组织进程五、查看进程六、通过系统调用获取进程标示符七、通过系统调用创建进程- fork初始7.1 fork函数创建子进程7.2 fork 之后通常要用 if 进行分流 八、进…

基于Springboot框架的学术期刊遴选服务-项目演示

项目介绍 本课程演示的是一款 基于Javaweb的水果超市管理系统&#xff0c;主要针对计算机相关专业的正在做毕设的学生与需要项目实战练习的 Java 学习者。 1.包含&#xff1a;项目源码、项目文档、数据库脚本、软件工具等所有资料 2.带你从零开始部署运行本套系统 3.该项目附…

2. K8S集群架构及主机准备

本次集群部署主机分布K8S集群主机配置主机静态IP设置主机名解析ipvs管理工具安装及模块加载主机系统升级主机间免密登录配置主机基础配置完后最好做个快照备份 2台负载均衡器 Haproxy高可用keepalived3台k8s master节点5台工作节点(至少2及以上)本次集群部署主机分布 K8S集群主…

Linux第105步_基于SiI9022A芯片的RGB转HDMI实验

SiI9022A是一款HDMI传输芯片&#xff0c;可以将“音视频接口”转换为HDMI或者DVI格式&#xff0c;是一个视频转换芯片。本实验基于linux的驱动程序设计。 SiI9022A支持输入视频格式有&#xff1a;xvYCC、BTA-T1004、ITU-R.656&#xff0c;内置DE发生器&#xff0c;支持SYNC格式…

物联网领域的MQTT协议,优势和应用场景

MQTT&#xff08;Message Queuing Telemetry Transport&#xff09;作为轻量级发布/订阅协议&#xff0c;凭借其低带宽消耗、低功耗与高扩展性&#xff0c;已成为物联网通信的事实标准。其核心优势包括&#xff1a;基于TCP/IP的异步通信机制、支持QoS&#xff08;服务质量&…