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个... 自媒体新手如何快速涨粉?这5个技巧让你少走弯路! 嗨,我是小融。 最近很多刚入门自媒体的朋友问我,怎...
乌兰察布市财政局关于黄金领域非... 乌兰察布市财政局关于黄金领域 非法金融活动风险提示 近期,黄金价格波动频繁,市场热度持续攀升,各类假...
一只鸡蛋架“直发”俄罗斯 无锡... (来源:无锡新传媒) 转自:无锡新传媒 一只3D打印塑料鸡蛋架,成为无锡国际邮件互换局正式开通运营后...
武汉楼市开启红五月 新房成交量... 原标题:武汉楼市开启红五月 数据爆表,新房成交量较去年同期翻番 武汉城建未来中心项目营销中心现场来...
一家精神病院竟现身A股公司前十... 5月8日,有投资者发现,盛通股份前十大股东名单中,竟出现了一家精神病院的身影。这家精神病院全称为“上...
真的老了!哈登心魔难除 骑士还... 哈登又拉胯了。 刚刚过去的两场东部半决赛,骑士都输的相当狼狈,而哈登的发挥更是灾难级的。 半决赛G1...
精神病院通报成上市公司前十大股... 近日,上市公司盛通股份发布一季报,披露了前十大股东名单。其中,一家名为“上饶市广丰区十五岭山精神病医...
天溯计量发布年报 上市首年检测... 转自:中国经营网 文 近日,计量检测机构天溯计量(301449.SZ)发布了2025年年度报告。年...
原创 全... 美伊真要停火了? 一页纸协议让全球油价闪崩! 就在今天,全球市场被一条消息炸开了锅。美国白宫觉得,他...
百信银行业绩:26Q1净利润大... 4月底,中信百信银行股份有限公司(下称“百信银行”)2025年财报及2026年一季度报接连披露—— ...
美光科技股价单周飙升38% 市... 【CNMO科技消息】受全球内存芯片短缺影响,美光科技股价本周大幅上涨。截至周五收盘,美光股价报746...
江西一精神病院炒股,炒成上市公... 近日,上市公司盛通股份(002599.SZ)发布一季报,披露了前十大股东名单,其中一家名为“上饶市广...
专访中国太保副总裁俞斌:从“+... 拥抱AI(人工智能),不再是保险行业的“选择题”,而是关乎企业生存与发展的“必答题”,更是企业决胜未...
多平台优化算法:美团取消超时扣... 图片来源:界面图库 5月8日,网信中国发布消息称,生活服务类平台算法治理已取得初步成效,美团、淘宝、...
原创 2... 2025年,国内系统重要性银行名单正式公布。这是我国金融体系的核心支柱,一共21家银行入选,它们是维...
东海县供销总社:“供销+龙头企... 近日,东海县供销合作总社鼎味泰直营店正式开业。作为东海县供销系统打造的新型社企便民服务网点,该门店的...
原创 阿... 深夜,一家零食店铺的客服后台弹出一条消息:“我上次买的芒果干,这次想换个不那么酸的口味,再帮我推荐几...
和平湾全新项目前瞻 负公摊、唯... 在沈阳,如果想在主城核心区域找一块容积率低于1.5的住宅用地,难度有多大? 过去三年,沈阳主城核心区...
精神病院与国际投行高盛同在 盛... 近日,盛通股份(002599.SZ)发布一季报,其前十大股东名单中,第九位为“上饶市广丰区十五岭山精...