javascript单线程
javascript的单线程,与它的用途有关。作为浏览器脚本语言,javascript的主要用途是与用户互动,以及操作dom。这决定了它只能是单线程,否则会带来很复杂的同步问题。比如,假定javascript同时有两个线程,一个线程在某个dom节点上添加内容,另一个线程删除了这个节点,这时浏览器应该以哪个线程为准?所以,为了避免复杂性,从一诞生,javascript就是单线程,这已经成了这门语言的核心特征,将来也不会改变。
队列任务
单线程就意味着,所有任务需要排队,前一个任务结束,才会执行后一个任务。如果前一个任务耗时很长,后一个任务就不得不一直等着。
异步事件驱动
浏览器中很多行为是异步(asynchronized)的,例如:鼠标点击事件、窗口大小拖拉事件、定时器触发事件、xmlhttprequest完成回调等。当一个异步事件发生的时候,它就进入事件队列。浏览器有一个内部大消息循环,event loop(事件循环),会轮询大的事件队列并处理事件。例如,浏览器当前正在忙于处理onclick事件,这时另外一个事件发生了(如:window onsize),这个异步事件就被放入事件队列等待处理,只有前面的处理完毕了,空闲了才会执行这个事件。
event loop
javascript是单线程的,但浏览器不是单线程的
浏览器至少会有以下一些进程
1.浏览器 gui 渲染线程
2.javascript 引擎线程
3.浏览器定时触发器线程
4.浏览器事件触发线程
5.浏览器 http 异步请求线程
因为 javascript 引擎是单线程的,所以代码都是先压到队列,然后由引擎采用先进先出的方式运行。事件处理函数、timer 执行函数也会排到这个队列中,然后利用一个无穷回圈,不断从队头取出函数执行,这个就是event loop。
总结一句话,js是单线程的,但是浏览器是多线程的,遇到异步的东西都是由浏览器把异步的回调放到event loop中,js线程不繁忙的时候,再去读取event loop
定时器原理
定时器的用法
settimeout(fn, delay)
setinterval(fn, delay)
fn是函数也可以是字符串,delay是延迟的时间,单位是毫秒
有以下要注意的
1.fn虽然可以是字符串,但是一直都不推荐这么使用
2.fn里面的函数如果有this,执行的时候this会指向到window上面去
如果理解好了js单线程和event loop,定时器的原理也很好理解了
如果设置了定时器,当到了延迟时间,浏览器会把延迟执行的事件放到event loop里面,当时间到了,如果js线程空闲就会执行了(所以定时器的精度不准啊)
看有文章介绍说settimeout和setinterval对函数一直轮询的区别,代码如下
复制代码 代码如下:
settimeout(function(){
settimeout(arguments.callee,100)
},100)
setinterval(function(){},1000)
文章大概的意思是settimeout是在回调函数执行后才启动下一次定时器,所以肯定是有间隔的执行,而setinterval是一直都在执行,如果碰到了js线程一直执行,可能就会在event loop里面加多个回调,当js线程不忙的时候,会一下子执行多个
经过测试发现无论在ie,ff,chrome,opera,safari下,setinterval都是按一定间隔来的
测试代码如下
复制代码 代码如下:
setinterval(function(){
xx.innerhtml=xx.innerhtml+1;
},100);
for(var i=0;i xx.offsetwidth
} settimeout(function(){
debugger;
},10)
断点的时候还是只打印出了1个1
定时器的精度问题
因为js单线程原因,如果遇到繁忙,定时器肯定不准,而且肯定是越来越长的,这个好像无解啊,无解啊
还有一个精度问题就是最小间隔settimeout(fun,0)
在js线程不忙的时候,也不可能0秒后马上执行,总有个最小间隔,每个浏览器还各不一样,这个未做测试
我看一篇文章中说的是w3c的标准,定时器的最小时间执行是4ms,找不到出处无从考证呀!!!
定时器相关的一些优化
在做定时器的时候还是可以有一些优化的
1.比如如果绑定window.onresize,在浏览器缩放的时候,该触发的非常频繁,所以可以延迟执行,当下次执行到的时候clear掉,减少频繁执行
伪代码如下
复制代码 代码如下:
var timer;
function r(){
cleartimeout(timer);
timer = settimeout(function(){
//do something
},150);
}
2.在滚动条往下拉的时候也是一下,比如图片的lazyload,也应该有一个定时器,避免过多的计算
3.当有多个地方需要定时器的时候,可以合并成一个定时器,时间间隔以最小的那个为准,然后需要执行的回调函数往数组里面塞,当到了时间间隔,遍历数组执行即可
一个小demo
复制代码 代码如下:
0
0
0
0
>http://static.paipaiimg.com/paipai_h5/js/ttj/zepto.min.js>>
小伙伴们是否对javascript定时器有所了解了呢,如有疑问给我留言吧。