服务器之家:专注于服务器技术及软件下载分享
分类导航

node.js|vue.js|jquery|angularjs|React|json|js教程|

服务器之家 - 编程语言 - JavaScript - js教程 - JS异步的执行原理和回调详解

JS异步的执行原理和回调详解

2022-02-13 17:24Zxinxxxx js教程

这篇文章主要给大家介绍了关于JS异步的执行原理和回调的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

一、JS异步的执行原理

  我们知道JavaScript是单线程的,而浏览器是多线程的。单线程执行任务需要一个个排队进行,假如一个任务需要很长时间执行(像ajax需要较长时间),会直接导致无响应,后面的任务一直在等待执行。这时候就需要用到异步。

  想了解异步,首先我们要知道浏览器有最基本的三个常驻线程: JS引擎线程,事件触发线程,GUI渲染线程。

  其中JS引擎线程和事件触发线程共同构成了一种事件循环机制,而GUI渲染线程与JS引擎是互斥的,当JS引擎执行时GUI线程会被挂起,GUI更新保存在一个队列中,当JS引擎空闲时,立即被执行。

  我们从它的事件循环机制解析:

JS异步的执行原理和回调详解

  JS引擎线程中分为同步和异步任务:

    1.同步任务全部通过主线程执行,形成执行栈。

    2.当有异步任务时交给异步进程(WebAPIs):包含事件触发线程或者定时器线程等处理,形成任务队列。

    3.当执行栈中的任务全部处理完成,主线程为空闲的时候,会从任务队列中提取任务到执行栈中执行。

  通俗来说,JavaScript除了主线程之外还存在一个任务队列,任务队列存放需要异步执行的内容,执行完主线程后,就会不断循环扫描执行任务队列的任务,直至队列清空。

画解:

JS异步的执行原理和回调详解

  如图小明因为学习耗时长会,如果没做完就会一直无法玩DNF游戏了,就把学习放到了异步任务队列中,等玩完游戏(主线程)再学习(任务队列)。期间母亲添加学习事件(DOM事件),小明每完成一个学习任务就看看还有啥任务(循环扫描),直至最后做完.

  下面再看一个例子(浏览器刷新不断点击按钮):

  let myData = null
  //ajax请求
  function ajax() {
  //腾讯新冠实时数据接口,仅做学习
  axios.get("https://api.inews.qq.com/newsqa/v1/query/inner/publish/modules/list?modules=chinaDayList,chinaDayAddList,nowConfirmStatis,provinceCompare")
   .then(data => {
   console.log("ajax返回成功");
   myData = data.data
   console.log(myData);

   })
   .catch(error => {
   console.log("ajax返回失败");
   })
  }
  console.log(myData);
  ajax()
  setTimeout(() => {
  console.log("定时器");
  }, 2000);
  console.log(myData);
  const btn = document.querySelector("button")
  btn.onclick = () => {
  console.log("点击了");
  }

null
null
ajax返回成功
Object
点击了
定时器
点击了

  可以看到,console在主线程中是同步执行的,先执行,而在主线程外的任务队列,存放着异步执行的内容,这里是setTimeout,ajax和DOM事件,按照任务队列顺序执行(循环扫描队列)。

  为什么要循环扫描呢?

  通过点击事件可以看出,当用户进行交互时(点击事件,滚动事件,窗口大小变化事件等),会向事件循环中的任务队列添加新事件,然后等待执行,所以需要循环扫描。

二、JS异步中的回调

  既然异步都是放在最后的任务队列执行,那么我们很多逻辑就难以实现,这时候我们需要处理这种异步逻辑,最常用的方式是回调――回头调用。

回调函数:简单来说就是,函数A中传入函数B作为参数时,函数B即为A函数执行的回调函数。回调有嵌套回调和链式回调两种。

  下面是回调的一个简单用法:

   let myData = null
   console.log(myData);
   setTimeout(() => {
    console.log("定时器");
   }, 2000);
   const btn = document.querySelector("button")
   btn.onclick = () => {
    console.log("点击了");
   }
   let name = "张三"
   function hr(callback) {
    setTimeout(() => {
     console.log(`我是${name}`);
     callback();
    }, 2001);
   }
   console.log(myData);
   function gj() {
    console.log(`${name}你好,我是李四,认识一下吧`);
   }
   hr(gj)

null
null
点击了
定时器
我是张三
张三你好,我是李四,认识一下吧
点击了

  很明显的看到,当我们函数需要用到数据的时候就用到了回调,这里用到的是异步回调。

  回调虽然是解决异步常用的方法,可是伴随着JS日益复杂的需求。同步异步需要越来越多的回调实现逻辑。同异步的混杂和过多的回调嵌套和缩进使得代码变得难以解读和维护,形成“回调地狱”。

JS异步的执行原理和回调详解

  我们看一个例子:

const verifyUser = function(username, password, callback){
  dataBase.verifyUser(username, password, (error, userInfo) => {
    if (error) {
      callback(error)
    }else{
      dataBase.getRoles(username, (error, roles) => {
        if (error){
          callback(error)
        }else {
          dataBase.logAccess(username, (error) => {
            if (error){
              callback(error);
            }else{
              callback(null, userInfo, roles);
            }
          })
        }
      })
    }
  })
};

大多数人光是看到上面的代码就感受到了脑子冻结的滋味,如果一个项目里拥有上百个这样的代码块,过一段时间,我相信连编写他的人都会头疼。来到自己的项目就像是来到了地狱。

  最主要的是,与此同时回调还存在信任问题,他把执行控制权交给了某个第三方(比如ajax)。为了解决信任问题,我们必须在程序写各种逻辑来解决回调带来的信任问题。

  ・调用过早

  ・调用过完

  ・调用次数过多过少,没有把需要的参数成功传给回调函数,

  ・可能出现的错误被吞。

  可以发现写特定逻辑来解决特定的信任问题,已经使得难度大于本身应用价值了,还会造成代码冗杂,可读性差等问题。

  综上:回调解决异步存在缺陷:

     1)不符合人对任务处理的逻辑思维

     2)回调带来的信任问题。

  面对回调日益明显的弊端,ES6更新了Promise用来解决异步问题。下一篇写ES6――Promise。

总结

到此这篇关于JS异步的执行原理和回调的文章就介绍到这了,更多相关JS异步执行原理回调内容请搜索服务器之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持服务器之家!

原文链接:https://blog.csdn.net/Zxinxxx/article/details/114444890

延伸 · 阅读

精彩推荐
  • js教程JS实现一个秒表计时器

    JS实现一个秒表计时器

    这篇文章主要为大家详细介绍了JS实现一个秒表计时器,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...

    yiran010100110152022-02-13
  • js教程JS 的 六种打断点的方式,你用过几种?

    JS 的 六种打断点的方式,你用过几种?

    Debugger 是前端开发很重要的一个工具,它可以在我们关心的代码处断住,通过单步运行来理清逻辑。而 Debugger 用的好坏与断点打得好坏有直接的关系。...

    神光的编程秘籍7932021-12-16
  • js教程mapboxgl实现带箭头轨迹线的代码

    mapboxgl实现带箭头轨迹线的代码

    这篇文章主要介绍了mapboxgl实现带箭头轨迹线的代码,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下...

    GIS兵器库9322021-12-27
  • js教程JavaScript canvas实现跟随鼠标移动小球

    JavaScript canvas实现跟随鼠标移动小球

    这篇文章主要为大家详细介绍了JavaScript canvas实现跟随鼠标移动小球,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一...

    清静清源10522022-01-20
  • js教程利用 JavaScript 实现并发控制的示例代码

    利用 JavaScript 实现并发控制的示例代码

    这篇文章主要介绍了利用 JavaScript 实现并发控制的示例代码,本文通过实例代码给大家介绍的非常想详细,对大家的学习或工作具有一定的参考借鉴价值,需...

    descire3902021-12-23
  • js教程微信小程序自定义胶囊样式

    微信小程序自定义胶囊样式

    这篇文章主要为大家详细介绍了微信小程序自定义胶囊样式,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...

    四曦11512021-12-21
  • js教程在HTML中使用JavaScript的两种方法

    在HTML中使用JavaScript的两种方法

    这篇文章主要介绍了在HTML中使用JavaScript的两种方法,帮助大家更好的理解和制作网页,感兴趣的朋友可以了解下...

    itbsl9432021-12-18
  • js教程javascript实现随机抽奖功能

    javascript实现随机抽奖功能

    这篇文章主要为大家详细介绍了javascript实现随机抽奖功能,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...

    彴兖7142021-12-23