jquery domReady实现原理

function bindReady(){

if ( readyBound ) return;

readyBound = true;

// Mozilla, Opera and webkit nightlies currently support this event

if ( document.addEventListener ) {

// Use the handy event callback

document.addEventListener( "DOMContentLoaded", function(){

document.removeEventListener( "DOMContentLoaded", arguments.callee, false );

jQuery.ready();

}, false );

// If IE event model is used

} else if ( document.attachEvent ) {

// ensure firing before onload,

// maybe late but safe also for iframes

document.attachEvent("onreadystatechange", function(){

if ( document.readyState === "complete" ) {

document.detachEvent( "onreadystatechange", arguments.callee );

jQuery.ready();

}

});

// If IE and not an iframe

// continually check to see if the document is ready

if ( document.documentElement.doScroll && typeof window.frameElement === "undefined" ) (function(){

if ( jQuery.isReady ) return;

try {

// If IE is used, use the trick by Diego Perini

// http://javascript.nwbox.com/IEContentLoaded/

document.documentElement.doScroll("left");

} catch( error ) {

setTimeout( arguments.callee, 0 );

return;

}

// and execute any waiting functions

jQuery.ready();

})();

}

// A fallback to window.onload, that will always work

jQuery.event.add( window, "load", jQuery.ready );

}

实现思路如下:

将Webkit与Firefox同等对待,都是直接注册DOMContentLoaded事件,但是由于Webkit是在525以上版本才引入的,因此存在兼容性的隐患。

对于IE,首先注册document的onreadystatechange事件,经测试,该方式与window.onload相当,依然会等到所有资源下载完毕后才

 

触发。

之后,判断如果是IE并且页面不在iframe当中,则通过setTiemout来不断的调用documentElement的doScroll方法,直到调用成功则

 

出触发DOMContentLoaded

jQuery对于IE的解决方案,使用了一种新的方法,该方法源自http://javascript.nwbox.com/IEContentLoaded/。 它的原理是,在

 

IE下,DOM的某些方法只有在DOM解析完成后才可以调用,doScroll就是这样一个方法,反过来当能调用doScroll的时候即 是DOM解析

 

完成之时,与prototype中的document.write相比,该方案可以解决页面有iframe时失效的问题。此外,jQuery 似乎担心当页面处于

 

iframe中时,该方法会失效,因此实现代码中做了判断,如果是在iframe中则通过document的 onreadystatechange来实现,否则通

 

过doScroll来实现。不过经测试,即使是在iframe中,doScroll依然有效。

 

posted @ 2017-03-06 16:40  sungang  阅读(430)  评论(0编辑  收藏  举报