DOM加载

前沿

处理页面文档加载的时候,我们遇到一个难题,就是使用window.onload这种将所有内容加载后(包括DOM文档结构,外部脚本、样式,图片音乐等)这样会导致在长时间加载页面的情况下,JS程序是不可用的状态。而JS其实只需要HTML DOM文档结构构造完毕之后就可以使用了,没必要等待诸如图片音乐和外部内容加载。


一.问题所在

首先了解一下浏览器加载的顺序:

1.HTML解析完毕;

2.外部脚本和样式加载完毕;

3.脚本在文档内解析并执行;

4.HTML DOM完全构造起来;

5.图片和外部内容加载;

6.网页完成加载。

 

PS:这里要了解一个问题,1-4的加载是极快的,一刹那而已。而第5条,根据网速和内容的多少各有快慢,但总体上如果有图片和外部内容的话,比1-4条加起来都要慢很多。

PS:并且JSdocument.getElementById这些只需要1-4条加载完毕后方可执行,并不需要加载第5条,所以,我们需要一种可以代替window.onload的更加快捷的加载方案。

前台测试代码:

<body>
<div id="box">box</div>
<img
src="http://h.hiphotos.baidu.com/album/s%3D1600%3Bq%3D100/sign=0686e4a05982b2b7a39f3
dc2019df09e/d01373f082025aaf03cd026ffbedab64024f1a92.jpg"></img

</body>

 //传统的DOM加载
window.onload = function () {
	var box = document.getElementById('box');         
	alert(box.innerHTML);
};

//PS:如果有图片,那么图片加载完毕后,方可执行onload里面的内容

二、解决方案

   非IE浏览器提供了一种加载事件:DOMContentLoaded事件,这个事件可以在完成HTML DOM结构之后就会触发,不会理会图像音乐、JS文件、CSS文件或其他资源是否已经下载完毕。

目前支持DOMContentLoaded事件浏览器有:IE9+FirefoxChromeSafari 3.1+Opera 9+都支持。

//DOMContentLoaded事件加载
if (document.addEventListener) {				//DOM结构加载完毕
	addEvent(document, 'DOMContentLoaded', function () {
		var box = document.getElementById('box');
		alert(box.innerHTML);
	});
} 
<span style="font-size:14px;color:#333333;"><span style="font-family:宋体;">//PS:如果有图片,先执行节点操作的内容,然后再缓缓加载图片,也就是说,当DOM树结构加载完毕后即可执行了</span></span>
<span style="font-size:14px;color:#333333;"><span style="font-family:宋体;"></span></span>
   

IE不支持DOMContentLoaded事件,可以采用方式:创建空script标签,属性拥有defer这个属性就是定义需要加载完毕后执行,然后待onreadystatechangecomplete时,表示DOM结构加载完毕了,再执行。

//IE浏览器加载
document.write("<script id=\"ie_onload\" defer=\"defer\" src=\"javascript:void(0)\">
<\/script>");
var script = document.getElementById("ie_onload");
script.onreadystatechange = function () {
	if (this.readyState=='complete') {
		var box = document.getElementById('box');
		alert(box.innerHTML);
	}
};
   

   另外,在IE浏览器如果网页上有<iframe>加载另一个网页,我们发现IE浏览器还需要加载完毕iframe所有的内容才可以执行。而非IE浏览器的DOMContentLoaded事件则还是DOM加载完毕后就执行了,在这里我们就发现IE的这种方式并不完美,当然,如果页面没有iframe的话就够用了。


  在IE中,任何DOM元素都有一个doScroll 方法,无论它们是否支持滚动条。为了判断DOM树是否建成,我们只看看documentElement是否完整就是,因为,它作为最外层的元素,作为 DOM树的根部而存在,如果documentElement完整的话,就可以调用doScroll方法了。当页面一加载JS时,我们就执行此方法,当然要 如果documentElement还不完整就会报错,我们在catch块中重新调用它,一直到成功执行,成功执行时就可以调用 fn方法了
//使用doScroll()来判断DOM加载完毕
var timer = null;
timer = setInterval(function () {
	try {
		document.documentElement.doScroll('left');
		var box = document.getElementById('box');
		alert(box.innerHTML);
	} catch (ex) {};
});


由此,我们可以结合一下上面两种方案,做一个兼容的函数方便调用。

function addDomLoaded(fn) {
	if (document.addEventListener) {			//W3C
		addEvent(document, 'DOMContentLoaded', function () {
			fn();
			removeEvent(document, 'DOMContentLoaded', arguments.callee);
		});
	} 
	else {								//IE
		var timer = null;
		timer = setInterval(function () {
			try {
				document.documentElement.doScroll('left');
				fn();
			} catch (ex) {};
		});
	}
}

addDomLoaded(function () {
	var box = document.getElementById('box');
	alert(box.innerHTML);
});
//PS:有效,DOM加载完毕后执行了节点操作,并且后面才加载完毕图片

三、总结:

虽然以上对于主流浏览器和主流浏览器的版本已经非常够用了,但还有几个小细节我们需要了解一下。Opera8之前不支持,webkit引擎浏览器525之前不支持,Firefox2有严重bug。
对于非IE,又不支持DOMContentLoaded,可以直接用传统的window.onload来执行,因为目前来说这种浏览器基本灭绝了,也可以document.readyState轮询,直到完毕。










  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值