HTML加载顺序
1. html文档中的所有dom标签会被按顺序加载包括css、js的内和外引用,在所有浏览器中都是如此。
1.1 仅有一个特殊情况:当使用document.write("js外引用地址");引用js时,IE不会去及时的加载和执行所引用的js。
解决方案:因为dom是一个标签一个标签加载,为了使后面的js代码使用到了这个外部js内容是不会找不到,可以将这段引用js的输出语句单独写在一个语句块中<script type="text/javascript">******</script>保证它一定被使用。
<span style="font-size:14px;"><script type="text/jscript">
document.write('<script type="text/javascript" src="test.js"></script>
</script></span>
1.2 也就是说html是同步加载(阻塞加载)的,前面的没有加载完,后边的不会被加载。比如:前面js写了个alert();那么你不去点这个alert的话,后面的所有元素都会在那等着被解析。
1.3 特别提醒:html文档并不十分严谨,即使你在<html></html>外面写div,js也是一样能被加载。
onload事件
1. onload是window对象的事件,它在窗口中所有东西都加载完时会被触发。记住它是窗口(window)中所有东西而不是文档(document),意味着如果你嵌入了<iframe>或者使用了<frameset>,而其中引用了很大的图片,此事件也会等图片完全加载了才去执行。
2. 当你有需要在window中所有东西都加载完时才去执行的js代码,你可以使用它。但大部分情况下,你的js不需要也不想等window全部加载完。
为什么需要ready()?
1.1 从目的来说,使用ready()函数的原因是你需要一个jquery环境来放置JQ相关代码;JQ团队让你将JQ代码放入一个ready()函数中,他们来保证你写的JQ代码一定能找得到JQ.js中的东西。
1.2 那么我们来看看他们是怎么写这个方法的:
<span style="font-size:18px;">jQuery.ready.promise = function( obj ) {
if ( !readyList ) {
readyList = jQuery.Deferred();
// Catch cases where $(document).ready() is called after the browser event has already occurred.
// we once tried to use readyState "interactive" here, but it caused issues like the one
// discovered by ChrisS here: http://bugs.jquery.com/ticket/12282#comment:15
if ( document.readyState === "complete" ) {
// Handle it asynchronously to allow scripts the opportunity to delay ready
setTimeout( jQuery.ready );
// Standards-based browsers support DOMContentLoaded
} else if ( document.addEventListener ) {
// Use the handy event callback
document.addEventListener( "DOMContentLoaded", completed, false );
// A fallback to window.onload, that will always work
window.addEventListener( "load", completed, false );
// If IE event model is used
} else {
// Ensure firing before onload, maybe late but safe also for iframes
document.attachEvent( "onreadystatechange", completed );
// A fallback to window.onload, that will always work
window.attachEvent( "onload", completed );
// If IE and not a frame
// continually check to see if the document is ready
var top = false;
try {
top = window.frameElement == null && document.documentElement;
} catch(e) {}
if ( top && top.doScroll ) {
(function doScrollCheck() {
if ( !jQuery.isReady ) {
try {
// Use the trick by Diego Perini
// http://javascript.nwbox.com/IEContentLoaded/
top.doScroll("left");
} catch(e) {
return setTimeout( doScrollCheck, 50 );
}
// detach all dom ready events
detach();
// and execute any waiting functions
jQuery.ready();
}
})();
}
}
}
return readyList.promise( obj );
};</span>
总结其代码逻辑来说:
1. 概念:
1.1 理想的文档加载完==*.html中所有标签元素加载完 ;这包括js、css的内外引用,不(想)包括图片、iframe和frameset引用的html中所有内容..(其他没测试)。
1.2 准备好ready:JQ定义的一个时间点,不同浏览器不同,JQ非常想使他们统一到一个时间点,但事实并不是,总有些差别。
1.2 标准浏览器:IE9+、FireFox、Chrome、Opera、Safari;他们支持addEventListener事件监听和DOMContentLoaded事件。
2. 流程
a:如果是标准浏览器:先使用标准浏览器都支持的事件DOMContentLoaded来判断文档是否加载完,加载完就认为ready,执行程序员自定义的ready内容。最不济,使用window.onload来触发你的ready代码。
b:如果不是标准浏览器(就是IE低版本):那就用onreadystatechage来判断文档是否加载完,认为ready。最不济也用window.onload。
b继续,如果是顶层(也就是说不是frame之类的),那么就认为top.doScroll("left");不抛出异常时认为ready。
3. 事实上
JQ非常想要统一其函数在所有浏览器上的体验,包括ready,但是“人生之不如意,十之八九”。
举个例子,IE11支持DOMContentLoaded,那么他会在这个事件发生后执行你的ready,但是如果有iframe和frameset的话IE11默认frame中内嵌的html文档完毕之后才会触发顶级文档的DOMContentLoaded。而Chrome测试就是不等frame中内嵌的html文档加载就会触发。还是不一样。
4. 所以
写JQ方法,请把内容写在ready()中(大多数情况下合适,也可能不合适),JQ团队尽量保证所有浏览器的统一体验。如有不测风云,造成代码错误,请明白我上面说的这些逻辑,然后请自行调试…