JavaScript 单线程相关

简介: 众所周知,Javascript是单线程执行的,这也就是说:JavaScript在同一个时间上只能处理一件事。他不像C,Java等这些多线程的,可以开不同的线程去同时处理多件事情。 那么为什么别的语言都可以这么方便的去开多个线程去同时执行多个任务,JavaScript却不行呢? “天将降大任于斯人也,必先苦其心志,劳其筋骨,饿其体肤,空乏其身,行指乱其所为,所以动心忍性,曾益其所不能”                                                 --《孟子》 正是因为JavaScript背负着重大的使命,所以他只能默默的看着别人拥有多线程。

众所周知,Javascript是单线程执行的,这也就是说:JavaScript在同一个时间上只能处理一件事。他不像C,Java等这些多线程的,可以开不同的线程去同时处理多件事情。

那么为什么别的语言都可以这么方便的去开多个线程去同时执行多个任务,JavaScript却不行呢?

“天将降大任于斯人也,必先苦其心志,劳其筋骨,饿其体肤,空乏其身,行指乱其所为,所以动心忍性,曾益其所不能”
                                                --《孟子》

正是因为JavaScript背负着重大的使命,所以他只能默默的看着别人拥有多线程。他作为浏览器脚本语言,只要是在于和用户进行交互/处理前端数据/操作DOM。

以代码举例吧:

  function addDom(){
    var html = document.createElement("div");
    html.innerHTML = "This is div "+i;
    document.getElementById("myDiv").appendChild(html);
  }
  function deleteDom(){
    var myDiv = document.getElementById("myDiv");
    //if(myDin.childNodes[0]){
        myDiv.removeChild(myDin.childNodes[0]);
    //}
  }

注意:以上代码注释部分可以让逻辑变的更严谨,此处需要报错已证明观点,所以加以注释。

按照以上代码执行addDom函数是能正确的生成这么一个DOM的吧,即便是在addDom函数还在执行过程中去执行deleteDom,deleteDom也将会在addDom执行完后才执行,那么就能完整的走完一个DOM的添加和删除操作了,这正是因为JavaScript是单线程。

现在我们假如JavaScript可以有多线程,那么我们让一个线程执行addDom函数的时候,在addDom还在执行的过程中再去执行deleteDom,这时候将会再开一个线程来执行,这时候浏览器脑子短路了,它不知道以哪个为主了...如果先执行完了deleteDom,那么效果就是即报了错还没达到想要的效果。

综上所述,JavaScript确实不适用于多线程,为了交互,他只能独自忍受了(所以当报错不会继续向下执行的时候,各位就别喷JavaScript了,好好检查自己写的代码才是这时候该做的)。

事件列队和异步执行

既然JavaScript是单线程执行的,那么有很多事件需要执行的时候,肯定需要排好队一个个来的吧。接下来我们就扯扯事件队列和异步。

JavaScript的事件队列里就是编排着接下来将要被逐个执行的事件,只有当前一个任务被执行完了,才会接下来执行后面一个任务。当我们触发一个事件,那么这个事件会被加入到事件列表末尾,当然不能插队咯,毕竟都是良民呐~

以上就是对事件队列的简单介绍,那么异步又是怎么回事呢?

Javascript语言将任务的执行模式分成两种:同步(Synchronous)和异步(Asynchronous)。

同步就是按照事件队列的顺序,有条不紊的执行下来。

异步则不同,每个任务都可以有一或多个回调函数(callback),当前面那一个任务执行完后,不去执行下一个任务,先执行其本身存在的回调函数,而下一个任务不等前一个任务结束,自顾自的开始执行了,这时候执行顺序就不一样的,产生异步了。(注意:这里并不说明使用回调函数即产生异步哦)

贴上代码:

  var num = 0;
  function firstFn(){
      num +=1;
  };
  function secondFn(value){
      if(value === 1){
          console.log("The num is 1"); //The num is 1
      } else{
          console.log("The num still 0");
      }
  }
  firstFn();
  secondFn(num);

以上是能正常打印出来的,因为先执行了第一个函数,所以这时num已经被加1了,所以判断生效。

  var num = 0;
  function firstFn(){
      setTimeout(function(){
         num +=1; 
      },0);  // 这里我们用setTimeout做了异步处理
  };
  function secondFn(value){
      if(value === 1){
          console.log("The num is 1");
      } else{
          console.log("The num still 0"); //The num still 0
      }
  }
  firstFn();
  secondFn(num);

这时候,再这样执行就没用了。因为setTimeout将num++的事件放到了事件列表的末尾去了,second(num)是在它的前面,所以现在执行是打印 The num still 0。那么怎么证明该事件被放到最后了呢?看下面的代码:

  <div id="myDiv" onclick="addEvent()">click me</div>
  var num = 0;
  function firstFn(){
      setTimeout(function(){
         num +=1; 
      },0);
  };
  function secondFn(value){
      if(value === 1){
          console.log("The num is 1"); //The num is 1
      } else{
          console.log("The num still 0");
      }
  }
  firstFn();
  function addEvent(){
      secondFn(num);
  } 

当我们点击id为myDiv的div的时候,将触发click事件吧,该事件会调用addEvent函数吧,addEvent函数会在事件队列的末尾加入新的需要执行的事件吧。这时候我们点击该div,就会打印 The num is 1 了。

同理理解setInterval。

下面顺便贴一段有小伙伴提问过的问题代码:

正常的代码:

  var i = 3;
  for(;i>0;i--){
    console.log(i);  //打印顺序:3   2   1
  };

"不正常"的代码:

  var i = 3;
  for(;i>0;i--){
    setTimeout(function(){console.log(i);},0);  //打印顺序 0   0   0
  };

人家问的就是为什么都设置延迟时间为0了,打印出来的还是不正常的。这也是因为异步,当函数被执行的时候,i的值已经为0。

对于setTimeout的通常描述:给定一个回调及N毫秒的延迟,setTimeout将会在N毫秒后运行该回调。

所以大多数情况下,这个描述只能算大致正确,不能算完全正确。

而且setTimeout还附带了个隐藏的可大可小的坑(将由线程阻塞导致):

  var start = new Date;
  setTimeout(function(){
      var end = new Date;
      console.log("End Time: ",end - start," ms"); // 有几次打印的是End Time:  1001  ms ,这还是在没有其他阻塞的,只运行这一个事件的情况下出现偏差
  },1000);

所以上面那个描述的N毫秒将在某些情况下会出现偏差。
好了,就写这么多先吧。明天还得上班呢,今天算拖的很晚了...还是手机编辑的,本来打算睡觉,写一半没完成,心里怪难受的...
如理解有偏差,还望大家不吝指教,大家一起交流才能更好的进步。

相关文章
|
9月前
|
JavaScript 前端开发
js单线程、同步、异步
什么是单线程?同步、异步的产生?
83 0
|
9月前
|
前端开发 JavaScript
单线程JavaScript为何如此高效
什么是js执行机制 JavaScript 的执行机制指的是 JavaScript 代码在运行时的工作方式和顺序
47 0
|
9月前
|
存储 JavaScript 前端开发
人人都能看懂的JavaScript单线程的那点事
人人都能看懂的JavaScript单线程的那点事
|
存储 JavaScript 前端开发
单线程的 【JavaScript】 是如何管理任务的
单线程的 【JavaScript】 是如何管理任务的
47 0
单线程的 【JavaScript】 是如何管理任务的
|
前端开发 JavaScript API
🍉如何理解单线程的JavaScript及其工作原理|8月更文挑战
🍉如何理解单线程的JavaScript及其工作原理|8月更文挑战
131 7
🍉如何理解单线程的JavaScript及其工作原理|8月更文挑战
|
消息中间件 Web App开发 存储
浅谈浏览器架构、单线程js、事件循环、消息队列、宏任务和微任务
关键词:多进程、单线程、事件循环、消息队列、宏任务、微任务
浅谈浏览器架构、单线程js、事件循环、消息队列、宏任务和微任务
|
消息中间件 JavaScript 前端开发
深入理解Javascript单线程谈Event Loop
深入理解Javascript单线程谈Event Loop
149 5
|
消息中间件 Web App开发 移动开发
JS是单线程,你了解其运行机制吗?
JS是单线程,你了解其运行机制吗?
145 0
JS是单线程,你了解其运行机制吗?
|
Web App开发 JavaScript 前端开发
Javascript定时器(一)——单线程
可以从下面的代码中看到,第一个用setTimeout中的代码是死循环,由于是单线程,下面的两个定时器就没机会执行了。
Javascript定时器(一)——单线程
|
JavaScript 前端开发 程序员
好程序员web前端培训分享如何理解JS的单线程
好程序员web前端培训分享如何理解JS单线程,JS本质是单线程的。也就是说,它并不能像JAVA语言那样,两个线程并发执行。 但我们平时看到的JS,分明是可以同时运作很多任务的,这又是怎么回事呢? 首先,JS的代码,大致分为两类,同步代码和异步代码。
851 0