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了,打印出来的还是不正常的。
对于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毫秒将在某些情况下会出现偏差。
好了,就写这么多先吧。明天还得上班呢,今天算拖的很晚了...还是手机编辑的,本来打算睡觉,写一半没完成,心里怪难受的...
如理解有偏差,还望大家不吝指教,大家一起交流才能更好的进步。
JavaScript 扯几句单线程相关的更多相关文章
- JavaScript 单线程相关
众所周知,Javascript是单线程执行的,这也就是说:JavaScript在同一个时间上只能处理一件事.他不像C,Java等这些多线程的,可以开不同的线程去同时处理多件事情. 那么为什么别的语言都 ...
- JavaScript 的异步和单线程
问题 Q:下面的代码是否能满足sleep效果? var t = true; setTimeout(function(){ t = false; }, 1000); while(t){ } alert( ...
- JavaScript 对象 - 与属性的相关知识
function inherit(p){ if(p == null) throw TypeError(); if(Object.create) return Object.create(p); var ...
- JavaScript跨浏览器处理事件以及相关对象
主流的浏览器和IE浏览器在处理事件和事件对象上是有所区别的,我们一般会通过EventUtil进行封装,这样,就可以正常的跨浏览器处理事件了,本文的主要内容总结自<JavaScript高级程序设计 ...
- 0182 JavaScript执行机制:单线程,同步任务和异步任务,执行栈,消息队列,事件循环
以下代码执行的结果是什么? [结果是1 2 3 ] console.log(1); setTimeout(function () { console.log(3); }, 1000); console ...
- JavaScript深入浅出3-语句
慕课网教程视频地址:Javascript深入浅出 程序由语句组成,语句遵守特定语法规则 块 block {} 没有块级作用域 声明 var 异常 try catch finally 函 ...
- JavaScript中forEach的用法相关
首先说下JavaScript的forEach的标准格式. 为数组中的每个元素执行指定操作. array1.forEach(callbackfn[, thisArg]) 参数 定义 array1 必需. ...
- JavaScript中的DOM及相关操作
一.什么是DOM JavaScript由ECMAScript.DOM和BOM三部分组成,其中DOM代表描述网页内容的方法和接口,即文档对象模型(Document Object Model).在网页上, ...
- JavaScript常用語句
1.document.write(""); 输出语句2.JS中的注释为//3.传统的HTML文档顺序是: document->html->(head,body)4 ...
随机推荐
- OpenGL------三维变换
我们生活在一个三维的世界——如果要观察一个物体,我们可以:1.从不同的位置去观察它.(视图变换)2.移动或者旋转它,当然了,如果它只是计算机里面的物体,我们还可以放大或缩小它.(模型变换)3.如果把物 ...
- HDOJ1312<DFS>
题意: 给一张图,有墙,有路.问某人从起点开始,最多能走多少个格子. 思路: bfs;<水题> #include<iostream> #include<cstring&g ...
- u-boot添加一个hello命令
1.在common目录下建立一个cmd_hello.c文件 2.仿照/common/cmd_bootm.c文件修改,把cmd_bootm.c头文件复制过来 3.再复制do_bootm.U_BOOT_C ...
- FZU Problem 2213 Common Tangents
其实是不太好意思往博客上放的,因为是一道巨水的题,但是我却错了一次,没有判断重合,放上还是为了警示自己,尽量不要在水题上罚时 #include<iostream> #include< ...
- Qt5:无边框窗口拖动
在窗口程序中,无边框窗口程序一般需要特殊处理才能拖动 Qt中,要实现无边框窗口的拖动,需要重新实现 mousePressEvent 和 mouseMoveEvent 俩虚函数 void Widget: ...
- 深入浅出Ajax(三)
<html xmlns="http://www.w3.org/1999/xhtml"> <head runat="server"> &l ...
- Android Studio实现Service AIDL
Android Studio实现Service AIDL [日期:2015-01-02] 来源:Linux社区 作者:teenyboy [字体:大 中 小] 今天要开发过程中要用到AID ...
- C语言写的俄罗斯方块
源:C语言写的俄罗斯方块 2014年最后一天, 任天堂将风靡全球30年的经典游戏<<俄罗斯方块>>下架. 作为全球最畅销的游戏, 其移植版本遍布各个平台. 下面这个是我去年在5 ...
- 再探CRC(转)
源:http://hi.baidu.com/skystalker/item/228a263147f74e87f5e4ad8d 之前写了CRC16的程序,虽说能用,却不知其所心然,现在要用CRC32,重 ...
- .net remoting 实现通用消息处理窗口
.net remoting 实现通用消息处理窗口 实现机制是制作一个cmd窗口作为信息展示窗口,主程序将需要展示的信息抛出到cmd窗口显示,以此方式做到消息的展示. 以下是cmd窗口的代码,cmd窗体 ...