js学习9(同步和异步)
创始人
2025-05-29 14:16:30
0

同步和异步

### 前言:
不得不说,同步有同步的好处和坏处,异步有异步的好处坏处,各有利弊。

### 思量:
(1) 如果有一个操作很费时间,那么在同步执行的过程中,该操作就会阻断后面代码的执行,这在有时是不利的【这时可以考虑异步】
(2)如果某个操作建立在某个异步操作返回的结果基础上,此时如果异步操作还没完成这样就会导致这个操作无法正常运行【这时可以考虑同步】

js同步和异步 

### 前言:
(1) 像平时编写的js代码,都是从上往下依次执行的(不涉及到异步方法或代码),就是同步代码;
(2) 当然js用有些方法的执行过程是异步的,比如常见的setTimeout()、fetch()等等;


### 解决异步问题,实现异步编程:
## 方式1:回调函数
在异步函数执行完成后(或满足某一状态后),执行回调函数,这样就能确保回调函数一定在异步任务之后执行的
## 方式2:使用Promise(then)
Promise的resolve状态可以断定当前Promise是否完成,且then在resolved状态时被执行,从而保证了Promise之间的顺序
## 方式3:使用Promise(async&await)
async声明的异步函数,可以使用await来确保Promise处于resolved状态再执行其后的代码
## 方式4:使用Promise(all)
Promise.all([promise1,promise2,...])允许将多个Promise对象组成链,并按照顺序依次先后执行
## 方式5:使用生成器
生成器通过yield终止执行,同时提供next方法继续执行,形成一个终止再执行的驱动过程,同时会把异步任务按顺序加入线程,整个线程完成后按顺序返回结果

Promise

### 创建

new Promise((resolve,reject)=>{

        ...

})

### Promise链

Promise对象无论是then方法还是catch方法返回的依然是Promise对象,所以会形成Promise链。

### then和catch

当Promise成为resolved状态后,会执行then方法;

当Promise成为rejected状态后,会执行catch方法;
 

### resolve()和reject()

resolve([value])使当前Promise状态为resolved状态,且可以传value给下一个Promise;

reject(err)使当前Promise状态为rejected状态;

### 代码了解

const promise1 = new Promise((resolve, reject) => {

  // 异步函数
  setTimeout(() => {
    console.log(222);
    resolve('foo'); // 设为resolved状态
  }, 5000);
});
promise1.then((value) => {
  console.log(333,value); // 等待Promise中异步函数的完成
});

console.log(111,promise1); // 同步代码

### 更多学习

 Promise - JavaScript | MDN (mozilla.org)

 

async&await

### 简要:

async允许声明一个异步函数,返回的是一个Promise对象,await可以等待一个Promise对象的完成再执行其后代码

### 代码了解

async function f1() {

    return new Promise((resolve,reject)=>{

        setTimeout(() => {

            console.log(111);

            resolve();

        }, 5000);

    })

   

}

f1().then(()=>{

    console.log(222);

})

异步编程

 1. XMLHttpRequest

### 异步(默认情况)
xhr.open(method,url,async=true);
## 异步过程,可以通过onload和onerror监听状态进而使用回调处理
xhr.onload = function(){var data = JSON.parse(this.responseText);
}### 同步
xhr.open(method,url,async=false);
# 同步过程,会等请求完成再执行下面代码
var data = JSON.parse(xhr.responseText());### 说明
第一种默认异步请求,为保证能拿到数据并处理,使用了回调
第二种直接为同步请求,其后代码会等待请求响应完成后再执行,也是可以的

 2. setTimeout

### 异步执行(默认情况)
console.log(111);
// setTimeout是个异步方法
setTimeout(()=>{console.log(222);
},5000)
console.log(333);
## 结果:111 >> 333 >> 222### 使其同步执行,结果:111 >> 222 >> 333
## 方式1:回调
console.log(111);
function f1(callback) {setTimeout(() => {console.log(222);callback();}, 5000);
}
function f2() {console.log(333);
}
f1(f2);## 方式2:Promise(then)
console.log(111);
new Promise((resolve, reject) => {setTimeout(() => {console.log(222);resolve();}, 5000)
})
.then(()=>{console.log(333);
})## 方式3:Promise + async&await
console.log(111);
function f1() {return new Promise((resolve, reject) => {setTimeout(() => {console.log(222);resolve();}, 5000)})
}
// 使用async + await
(async function test() {await f1();    console.log(333)
})();### 总结:
(1) js中已经封装好的方法,一般用监听器回调的方式来解决异步问题;
(2) 自己编写异步代码,则可以使用Promise方式;

回调地狱

简要:

多个异步任务使用回调的方式来确保运行顺序,此时回调形成嵌套的多层缩进问题

 回调地狱代码(简略形式): 

function f1(callback) {setTimeout(() => {console.log(222);callback();}, 5000);
}
function f2(callback){setTimeout(() => {console.log(333);callback();}, 3000);
}
function f3() {console.log(444);
}f1(()=>{f2(()=>{f3();})
});

  解决改进 :

### 方式1:Promise(then)
function p1(){return new Promise((resolve,reject)=>{setTimeout(() => {console.log(111);resolve();}, 5000);})
} 
function p2(){return new Promise((resolve,reject)=>{setTimeout(() => {console.log(222);resolve();}, 3000);})
} 
function p3(){return new Promise((resolve,reject)=>{setTimeout(() => {console.log(333);resolve();}, 1000);})
} p1()
.then(()=>p2())
.then(()=>p3());## 方式2:Promise + async&await
function p1() {return new Promise((resolve,reject)=>{setTimeout(() => {console.log(111);resolve();}, 5000);})
}:
function p2() {return new Promise((resolve,reject)=>{setTimeout(() => {console.log(222);resolve();}, 5000);})
}
(async ()=>{await p1();await p2();console.log(333);
})();## 方式3:Promise.all()
p1 = new Promise((resolve,reject)=>{setTimeout(() => {console.log(111);resolve();}, 5000);
})
p2 = new Promise((resolve,reject)=>{setTimeout(() => {console.log(222);resolve();}, 5000);
})Promise.all([p1,p2])
.then(()=>{console.log(333);
});

相关内容

热门资讯

当对手都在做下沉 蜜雪冰城旗下... [ 今年5月,蜜雪集团跟巴西签署40亿元人民币的采购意向大单,其中大多数是咖啡豆。 ] 当星巴克、瑞...
新手必看!股指期货交易规则基础... 股指期货交易规则,看似复杂抽象,实则与我们的日常生活有着奇妙的共通之处。它就像一场精心编排的生活交响...
王登发履新茅台技开公司“一把手... 一则微信公众号发布的信息,披露了茅台集团旗下的技术开发公司“一把手”已换人。 近日,南都湾财社-酒水...
特斯拉机器人V3量产版亮相!马... 快科技7月27日消息,特斯拉的Optimus人形机器人V3量产版终于要来了!马斯克在最近的财报电话会...
原创 中... 在金融全球化的浪潮中,中国资本市场始终勇立潮头,不断探索前行。7月26日,中国资本市场学会成立大会暨...
报告:我国经济增长保持韧性 下... 央广网北京7月27日消息(记者 樊瑞)近日,中国金融四十人论坛(CF40论坛)发布《2025年第二季...
超6300亿元!A股银行“分红... 7月25日,成都银行完成权益分派股权登记,将于7月28日发放现金红利,这标志着A股上市银行2024年...
老铺黄金:2025年上半年单个... 7月27日晚,老铺黄金(HK06181)披露2025年中期业绩预告。预计2025年上半年实现销售业绩...
保险行业2025年上半年回顾与... 今天分享的是:保险行业2025年上半年回顾与未来展望 报告共计:59页 2025年上半年保险行业回顾...
数币App上新!消费者、商户两... 数字人民币试点持续推进,相关数字钱包手机应用程序功能也在优化中。7月21日,北京商报记者注意到,日前...
A股热点迭出,个股连续涨停!资... 近段时间以来A股市场整体走势较为强劲,上周以来在雅江概念集体上行的推动下涨势更为明显,主要指数不同程...
原创 印... 令人惊讶的是,印度人开始反思自身制造业的发展状况。印度经济学家帕纳加利亚指出,印度原本有机会在20年...
首创证券拟赴港上市,“A+H”... 首创证券在A股上市不足三年便启动赴港上市计划。近日,首创证券公告称,公司董事会已审议通过了公司拟发行...
肥东杨大爷要帮“儿子”还钱,银... “儿子”在外借了2万元还不上 “要债人”电话直接打了过来 还?还是不还? 7月6日 肥东县公安局梁园...
A股上周16家上市公司公布并购... 转自:扬子晚报 扬子晚报网7月27日讯(记者 范晓林 薄云峰)近段时间以来,A股市场并购重组活跃度持...
独家|某股份行改动零售业务关键... 在资产端信贷“投不动”(多家行零售信贷增速连续几个季度放缓、更有甚者个贷投放负增长)、负债端存款“定...
四川五日游报团指南及详细行程,... 四川,这片位于中国西南的神奇土地,以其独特的自然风光、丰富的文化遗产和诱人的美食而闻名遐迩。从成都的...
原创 中... 在2025年4月初,时任美国总统的特朗普正式启动了针对世界各国的关税战,旨在通过实施经济制裁来促进美...
牛市主升浪开启了?别急!珍惜布... 本周,A股市场上行,主要宽基指数都收获了或多或少的周涨幅,其中,科创50、微盘股涨幅居前。板块方面,...
公募二季报两大看点!港股配置逼... 本报(chinatimes.net.cn)记者栗鹏菲 叶青 北京报道 2025年公募基金二季报披露收...