javascript异步理解

javascript是单线程编程,意思就是javascript引擎一次只能执行一个语句。这样在出现长时间的请求的话,会阻塞主线程,为了不阻塞主线程,出现了javascript异步的概念。

javascript 同步异步理解:
设想这样的情景,你现在路过一个炸鸡店,闻着特别香,你实在抵挡不住炸鸡的诱惑,你就走上前,买了一份炸鸡。
同步的情况是:你点了一份炸鸡,然后你站到跟前等着啥也不干,炸鸡好了之后,你付钱给老板。
异步的情况是:你点了一份炸鸡,然后老板给你一张号码单,你把钱付了,玩着手机等,炸鸡好了之后,老板再叫你。

同步代码:

const makeChicken = () => {
  console.log('炸鸡好了')
}
console.log('我要吃炸鸡')
makeChicken()
console.log('给你炸鸡钱')
console.log('玩手机')

执行结果:

我要吃炸鸡
炸鸡好了
给你炸鸡钱
玩手机

异步代码:

const makeChicken = () => {
  setTimeout(() => {
    console.log('炸鸡好了')
  }, 2000)
}
console.log('我要吃炸鸡')
makeChicken()
console.log('给你炸鸡钱')
console.log('玩手机')

执行结果:

我要吃炸鸡
给你炸鸡钱
玩手机
炸鸡好了

异步实现逻辑:


image.png

事件循环、web APIs 和消息队列不是JavaScript引擎的一部分,它们是浏览器端的JavaScript运行环境或者Nodejs端的JavaScript运行环境的一部分。在Nodejs中,web APIs被C/C++ APIs替代。

当上面的代码加载在浏览器中,console.log('我要吃炸鸡'),被推到栈中然后当结束时从栈中被移除。然后,makeChicken()触发,然后它被推到了栈的顶部。

接下来setTimeOut( )函数被调用,所以它被推到了栈顶。setTimeOut( )有两个参数:一是回调,二是毫秒数。

这个setTimeOut( )方法在web APIs环境中开始了一个2秒的计时器。此时,setTimeOut( )执行完毕并被移出栈。之后,console.log('给你炸鸡钱'),被推到栈里,执行完后被移除。

同时,计时器到期了,现在这个回调被推到了消息队列。但是这回调不会被立即执行,这里就是事件循环插手的地方。

事件循环的作用是查看调用栈并确定调用栈是否为空。 如果调用栈为空,它会查看消息队列以查看是否有任何挂起的回调等待执行。

在上面的例子中,消息队列中有一个回调,并且此时调用栈已经空了。 因此,事件循环将回调推到栈顶部。

之后, console.log('炸鸡好了')被推到栈顶部,执行完后从堆栈中弹出。 此时,回调已执行完毕,因此被从栈中移除,程序最终完成。

异步的几种实现:
1)回调函数:
意思就是等到炸鸡好了再叫你。

function getYourFood() {
  console.log('给你炸鸡')
}

function getChicken(callback) {
  console.log('准备制作炸鸡')
  setTimeout(() => {
    console.log('炸鸡制作完成')
    callback()
  }, 2000)
}

getChicken(getYourFood)

执行结果:

我要炸鸡
准备制作炸鸡
炸鸡制作完成
给你炸鸡

2)promise
使我们可以主动的去执行后面的动作,而不是把回调函数给一个方法等在这个方法内部调用。

function makeChicken() {
  return new Promise(((resolve) => {
    console.log('准备制作炸鸡')
    setTimeout(() => {
      resolve('炸鸡成功')
      console.log('炸鸡制作完成')
    }, 2000)
    console.log('炸鸡制作中')
  }))
}
function getYourFood() {
  return new Promise(((resolve) => {
    resolve('炸鸡好了')
  }))
}
console.log('我要炸鸡')
makeChicken().then(() => {
  getYourFood().then((seccess) => {
    console.log(seccess)
  })
})

执行结果:

我要炸鸡
准备制作炸鸡
炸鸡制作中
炸鸡制作完成
炸鸡好了

3)async await
使异步代码看起来就是同步代码。我们添加一个关键字async让引擎知道哪个函数触发是异步的并且会返回一个promise,让我们使用await。

function makeChicken() {
  return new Promise(((resolve) => {
    console.log('准备制作炸鸡')
    setTimeout(() => {
      resolve('炸鸡成功')
      console.log('炸鸡制作完成')
    }, 2000)
    console.log('炸鸡制作中')
  }))
}
function getYourFood() {
  return new Promise(((resolve) => {
    resolve('炸鸡好了')
  }))
}
async function OrderChicken() {
  await makeChicken()
  const result = await getYourFood()
  console.log(result)
}
console.log('我要炸鸡')
OrderChicken()

运行结果:

我要炸鸡
准备制作炸鸡
炸鸡制作中
炸鸡制作完成
炸鸡好了

我觉得下面链接的有关异步的讲解特别好。

参考链接:
[译]理解异步JavaScript-事件循环https://mbd.baidu.com/newspage/data/landingshare?pageType=1&isBdboxFrom=1&context=%7B%22nid%22%3A%22news_9071904341435333508%22%2C%22sourceFrom%22%3A%22bjh%22%7D

[翻译]JavaScript异步进化史:Callbacks,Promises,Async/Await 上
https://mbd.baidu.com/newspage/data/landingshare?context=%7B%22nid%22%3A%22news_8754335365643315921%22%2C%22sourceFrom%22%3A%22bjh%22%7D&type=news

[翻译]JavaScript异步进化史之下篇
https://mbd.baidu.com/newspage/data/landingshare?context=%7B%22nid%22%3A%22news_9356479249406614478%22%2C%22sourceFrom%22%3A%22bjh%22%7D&type=news

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

推荐阅读更多精彩内容