Javascript定时器(三)——setTimeout(func, 0)

setTimeout(func, 0)可以使用在很多地方,拆分循环、模拟事件捕获、页面渲染等

一、setTimeout中的delay参数为0,并不是指马上执行

<script type="text/javascript">
        function delay1() {
            console.log(‘delay1‘);
        }
        function delay2() {
            console.log(‘delay2‘);
        }
        function delay3() {
            console.log(‘delay3‘);
        }

        delay1();
        setTimeout(function() {
            delay2();
        }, 0);
        delay3();
</script>

  用firefox的firebug可以查看到,并不是按照delay1,delay2,delay3这样打印的。

  

  由于JavaScript是单线程处理任务的,而setTimeout是异步事件,延时时间为0的时候,JavaScript引擎会把异步事件立刻放到任务队列里,而不是立刻执行,需要等到前面处于等待状态的事件处理程序全部执行完成后再调用它(JavaScript engines only have a single thread, forcing asynchronous events to queue waiting for execution)。JavaScript引擎的运作可以查看前面的文章《Javascript定时器(一)——单线程》

二、拆解循环

循环体中包含太多的操作和循环的次数过多都会导致循环执行时间过长,并直接导致锁死浏览器。如果循环之后没有其他操作,每次循环只处理一个数值,而且不依赖于上一次循环的结果则可以对循环进行拆解。

<script type="text/javascript">
  function delay() {
      for(var i=0; i<100000000; i++) {
                //logic
      }
    }
    delay();
</script>

使用setTimeout拆解循环,用firebug的console.log打印index值。

<script type="text/javascript">
        var index = 0;
        //使用setTimeout拆解循环
        function major() {
            if(index > 100000000) {
                return;
            }
            console.log(index);
            index++;
            setTimeout(function() {
                major();
            }, 0);
        }
        major();
</script>

三、模拟事件捕获

浏览器的 DOM 事件都是采用冒泡的方式,在实际的开发过程中可能存在需要事件捕获的需求,要求子元素的事件在父元素触发之后才能触发。

  <p>1、父元素的事件在子元素触发之后触发</p>
    <div id="parent">
        <a href="javascript:void(0)" id="child">冒泡</a>
        <p id="result1"></p>
    </div>

  1、一个a按钮

  2、一个p显示结果

  3、一个div作为父级元素

<script type="text/javascript">
        //父元素的事件在子元素触发之后触发
        function bubbling() {
            var child = document.getElementById(‘child‘),
                parent = document.getElementById(‘parent‘),
                result = document.getElementById(‘result1‘);
            child.onclick = function() {
                result.innerHTML += ‘child->‘;
            };

            parent.onclick = function() {
                result.innerHTML += ‘parent->‘;
            };
        }
</script>

  1、绑定a按钮的click事件,点击一下打印child

  2、绑定div按钮的click事件,点击一下打印parent

  3、显示结果如下:

  

在a按钮的click事件中加个setTimeout,就能做到子元素的事件在父元素触发之后触发了,这里也是JavaScript引擎在起作用。

    child.onclick = function() {
            setTimeout(function() {
                result.innerHTML += ‘child->‘;
            }, 0);
       };

四、页面渲染

浏览器中GUI渲染线程与JavaScript引擎是互斥的,所以当JavaScript执行时,浏览器就不会做任何的页面渲染。

<p>3、页面渲染,阻塞</p>
<a href="javascript:void(0)" id="vary">变换</a>
<div style="width:150px;height:150px;border:1px solid #c6c6c6" id="container"></div>

  1、点击变换按钮,将下面container的宽与高从150减到1。

  2、由于互斥的关系,效果感觉是一下子从150到1了,中间没有那个减小的过程。

<script type="text/javascript">
     //页面渲染
        function render() {
            var btn = document.getElementById(‘vary‘), container = document.getElementById(‘container‘);
            btn.onclick = function() {
              for (var i = 150; i > 0; i--) {
                container.style.height = i + ‘px‘;
                container.style.width = i + ‘px‘;
              }
            };
        }
</script>

  1、修改container的height与width会引发回流(reflow),一回流GUI渲染线程线程就会执行。

  2、由于互斥的原因,只有当JavaScript执行完毕后,才会做渲染。

  =》

用setTimeout后,就会出现从150到1的减小过程。

       function vary(i) {
                i--;
                console.log(i);
                container.style.height = i + ‘px‘;
                container.style.width = i + ‘px‘;
                if(i > 0) {
                    setTimeout(function() {
                        vary(i);
                    }, 0);
                }
            }
            btn.onclick = function() {
                var i = 150;
                vary(i);
            };

Tips:

这里顺便说下重绘(Repaint)与回流(reflow):

重绘(repaints):是一个元素外观的改变所触发的浏览器行为,例如改变vidibility、outline、背景色等属性。浏览器会根据元素的新属性重新绘制,使元素呈现新的外观。重绘不会带来重新布局,并不一定伴随回流。
回流(reflow):是更明显的一种改变,可以理解为渲染树需要重新计算。

导致 reflow 发生的一些因素:

  1. 调整窗口大小(Resizing the window)
  2. 改变字体(Changing the font)
  3. 增加或者移除样式表(Adding or removing a stylesheet)
  4. 内容变化,比如用户在input框中输入文字(Content changes, such as a user typing text in an input box)
  5. 激活 CSS 伪类,比如 :hover (IE 中为兄弟结点伪类的激活)(Activation of CSS pseudo classes such as :hover (in IE the activation of the pseudo class of a sibling))
  6. 操作 class 属性(Manipulating the class attribute)
  7. 脚本操作 DOM(A script manipulating the DOM)
  8. 计算 offsetWidth 和 offsetHeight 属性(Calculating offsetWidth and offsetHeight)
  9. 设置 style 属性的值 (Setting a property of the style attribute)

demo下载:

http://download.csdn.net/download/loneleaf1/7971185

参考资料:

http://heroicyang.com/2012/09/22/javascript-timer-in-depth/ 深入理解JavaScript定时器

http://heroicyang.com/2012/10/14/javascript-timer-and-event-in-depth/ 深入理解JavaScript定时器(续)

http://my.oschina.net/cupidooo/blog/149379 Web前端:解决浏览器页面回流(reflow)的几种方法

http://www.planabc.net/2009/04/13/reflow/ 影响 reflow 的因素及其优化

http://www.nowamagic.net/librarys/veda/detail/787 优化js脚本设计,防止浏览器假死

时间: 2024-10-22 10:20:39

Javascript定时器(三)——setTimeout(func, 0)的相关文章

Javascript定时器(二)——setTimeout与setInterval

一.解释说明 1.概述 setTimeout:在指定的延迟时间之后调用一个函数或者执行一个代码片段 setInterval:周期性地调用一个函数(function)或者执行一段代码. 2.语法 setTimeout: var timeoutID = window.setTimeout(func, delay, [param1, param2, ...]); var timeoutID = window.setTimeout(code, delay); timeoutID 是该延时操作的数字ID,

JavaScript下的setTimeout(fn,0)意味着什么?

原文:JavaScript下的setTimeout(fn,0)意味着什么? 近期在研究异步编程的我对于setTimeout之类的东西异常敏感.在SegmentFault上看到了一个问题<关于SetTimeout时间设为0时>:提问者读了一篇文章,原文解释setTimeout延迟时间为0时会发生的事情,提问者提出了几个文章中的几个疑点.读了那篇文章之后发现原文的作者对于setTimeout的理解和自己的认知有点出入,于是编写了相关测试的代码以求答案.最终编写了这篇文章. 本文内容如下: 起因 单

【编辑推荐】JavaScript下的setTimeout(fn,0)意味着什么?

近期在研究异步编程的我对于setTimeout之类的东西异常敏感.在SegmentFault上看到了一个问题<关于SetTimeout时间设为0时>:提问者读了一篇文章,原文解释setTimeout延迟时间为0时会发生的事情,提问者提出了几个文章中的几个疑点.读了那篇文章之后发现原文的作者对于setTimeout的理解和自己的认知有点出入,于是编写了相关测试的代码以求答案.最终编写了这篇文章. 本文内容如下: 起因 单线程的JavaScript setTimeout背后意味着什么 参考和引用

JavaScript下的setTimeout(fn,0)的作用,涨知识了

在工作学习中,我看到团队中的有些大神的代码中不时的出现setTimeout(fn,0):这样的写法,很好奇,这样写的作用是什么,在网上,经过一位网友无私耐心地阐述,我才对js单线程的机制有了更深的认识,此处我引上链接,与君共勉! http://www.cnblogs.com/silin6/p/4333999.html

Javascript 定时器篇 setTimeout和setInterval

标题定时器,让我想起了年代久远的VB,那时候也有个定时器,长的跟闹钟一样()相信跟我一样用过VB的人都不陌生.很基础的东西,但是作用却很大.记得那时候我还不太会用,忘记是用来做动画了,还是干嘛了.不过如今,已经入门编程语言了,那么就用CODE来说吧.今天我么要说的是WEB 前端技术 Javascript 的2大定时器.存活了很久,用处依旧大大大的~~~他们分别就是 setTimeout和setInterval. Javascript本身区分大小写,所以要注意 setTimeout和setInte

移动 Web 与 JavaScript 定时器

转载自:https://www.imququ.com/post/mobile_web_and_js_timer.html 在之前博客中,我曾经写过「PC 上的 Firefox.Chrome 和 Safari 等浏览器,都会自动把未激活页面中的 JavaScript 定时器(setTimeout.setInterval)间隔最小值改为 1 秒以上:而移动设备上的浏览器往往会直接冻结未激活页面上的所有定时器」.今天继续聊一聊 JavaScript 定时器与移动 Web 这个话题. 计时器 最简单的计

JavaScript: 再论setTimeout、setInterval。其第三个参数和this的讨论,超时嵌套和内存泄漏

最近用setTimeout.setInterval,因为要传入的函数要用到this,所以深入了解了一番! setTimeout和setInterval函数的第三个参数本来只是定义语言类型,后来在非IE浏览器下支持传递参数,并且在不同浏览器下支持的不同. 原来的setTimeout函数定义: var timeoutID = window.setTimeout(func, delay[, lang]); 在Chrome和FF下定义被修改: var timeoutID = window.setTime

Javascript 的定时器 setInterval,setTimeout,clearInterval

今天开通博客.来1个.哇哈哈哈~~ 今天本来想复习BOM的看到定时器也算DOM一种 ?(是这样吗).分享一下 参考源于:八神吻你 http://www.cnblogs.com/lmfeng/archive/2011/06/24/2089237.html 单次定时器即执行 1 //单次定时器即执行 2 3 setInterval(console.log(1),1000); //执行1次 4 5 //------------------------------------ 6 7 var a = f

JavaScript定时器原理分析

.header { cursor: pointer } p { margin: 3px 6px } th { background: lightblue; width: 20% } table { text-align: center; margin-top: 20px; margin-left: 10px; margin-bottom: 20px } a { cursor: pointer; text-decoration: none; color: gray } a:hover { text