domReady的实现
我们都知道JQ的 $(document).ready(fn) 方法。可以在页面准备就绪后才执行脚本,该方法相比传统的window.onload 事件,它的优势体现于onload事件是需要等到页面中所有资源都加载完毕后才会触发,而JQ的ready方法则会判断DOM树是否构建完毕。
onload相比较onreadystate事件的区别是,onreadystatechange事件是IE独有的,并且在IE11之后不再支持,该事件是IE浏览器为特定的需要判断资源加载的DOM元素指定的事件。
支持onreadystatechange事件的DOM元素必然有一个readyState属性,该属性的返回值,用于说明资源的加载情况。
一般而言,onreadystatechange事件更多用于Iframe的加载判断。
最后我们需要了解的是当页面包含iframe后,DOM树的生成,以及DOMContentLoaded事件的触发,onload事件的触发,其流程对于IE非IE是不同的。
一般来说:
IE :解析index页面 -> 解析iframe页面 -> 触发iframe的DOMContentLoaded事件 -> 触发iframe页面 onload事件 -> 触发Index页面的DOMContentLoaded事件 -> 触发index页面的onload事件。
!IE : 解析index页面 -> 触发index页面的DOMContentLoaded事件 -> 解析iframe页面 -> 触发iframe页面的DOMContentLoaded事件 -> 触发iframe的onload事件 -> 触发index页面的onload事件。
从这个流程,我们可以看出IE中,必须等待当前页面的iframe加载解析完毕,当前页面才能加载解析完毕,而在非IE中,iframe的加载与解析对当前页面来言更多的是异步执行。
下面是具体的代码:
1 (function(win){ 2 3 'use strict'; 4 5 var document = win.document, 6 readList = [], // 等待执行的函数堆栈 7 flag = false; 8 9 var removeEvent = function(){ 10 11 if(document.addEventListenner){ 12 window.removeEventListenner('load',handle,false); 13 }else if(document.attachEvent){ 14 window.detachEvent('onload',handle) 15 document.detachEvent('onreadystatechange',readyState); 16 }else{ 17 window.onload = null; 18 } 19 20 }, 21 handle = function(){ 22 23 if(!flag){ 24 25 while(readList.length){ 26 readList[0].call(); //执行函数 27 readList.shift(); //删除第一个数组元素 28 } 29 flag = true; 30 removeEvent(); 31 } 32 33 }, 34 readyState = function(){ 35 if(document.readyState == 'complete'){ 36 handle(); 37 } 38 }, 39 DOMContentloaded=function(){ 40 41 if(document.readyState == 'complete'){ 42 setTimeout(handle); // setTimeout 会使用最短时间,该时间不同系统并不一样。 43 }else if(document.addEventListenner){ 44 document.addEventListenner('DOMContentLoaded',fn,false); 45 window.addEventListenner('load',handle,false); 46 }else if(document.attachEvent){ 47 window.attachEvent('onload',handle); 48 document.attachEvent('onreadystatechange',readyState); //onreadystatechange 事件在页面中含有iframe的时候,它会等待iframe加载完毕才会触发。 49 50 if(self === self.top){ // 当页面不在iframe中则使用此种方式检测doScroll方法是否可用。如果再iframe中则用onreadstatechange事件进行判断。 51 (function(){ 52 try{ 53 document.documentElement.doScroll('left'); 54 }catch(e){ 55 setTimeout(arguments.callee,50); //arguments.callee 是对当前函数的引用。 56 return ; 57 } 58 handle(); 59 }()); 60 } 61 62 }else{ 63 window.onload = handle; 64 } 65 }, 66 ready = function(fn){ 67 readList.push(fn); // 加入待处理的堆栈中。 68 DOMContentloaded(); 69 }; 70 71 win.domReady = ready; 72 73 74 }(window));
代码调用:
1 domReady(function(){ 2 document.getElementById('box').innerHTML = (new Date().getTime() - date)/1000; 3 });
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
· TypeScript + Deepseek 打造卜卦网站:技术与玄学的结合
· 阿里巴巴 QwQ-32B真的超越了 DeepSeek R-1吗?
· 【译】Visual Studio 中新的强大生产力特性
· 10年+ .NET Coder 心语 ── 封装的思维:从隐藏、稳定开始理解其本质意义
· 【设计模式】告别冗长if-else语句:使用策略模式优化代码结构
2015-10-11 Ajax 学习 - 基础学习