当前位置: 首页 > 编程日记 > 正文

DOMContentLoaded 与onload区别以及使用

一、何时触发这两个事件?

1、当 onload 事件触发时,页面上所有的DOM,样式表,脚本,图片,flash都已经加载完成了。

2、当 DOMContentLoaded 事件触发时,仅当DOM加载完成,不包括样式表,图片,flash。

二、为什么要区分?

开发中我们经常需要给一些元素的事件绑定处理函数。但问题是,如果那个元素还没有加载到页面上,但是绑定事件已经执行完了,是没有效果的。这两个事件大致就是用来避免这样一种情况,将绑定的函数放在这两个事件的回调中,保证能在页面的某些元素加载完毕之后再绑定事件的函数。

当然DOMContentLoaded机制更加合理,因为我们可以容忍图片,flash延迟加载,却不可以容忍看见内容后页面不可交互。

这里又要牵扯到页面加载渲染的原理了:

1、加载样式表会阻塞外链脚本的执行

  一些Gecko和Webkit引擎版本的浏览器,包括IE8在内,会同时发起多个Http请求来并行下在样式表和脚本。但脚本不会被执行,直到样式被加载完成。在未加载完之前甚至页面也不会被渲染。但是在opera中样式的加载不会阻塞脚本的执行。

因此:目前通用的作法是把脚本和样式都以外链形式引入,甚至在jquery的官方文档中也是这样推荐的。对于大部分脚本来说,这样的脚本等待外链的机制还是有意义的,比如一些DOM和样式操作需要读取元素的位置,颜色等。这就需要样式先于脚本加载

检验方法:尝试强制使服务器端使style延迟一段时间才加载(甚至10秒),测试的结果是,在某些版本的Firefox,Chrome中最后一段脚本仍然是可以读出style的属性值(因为style始终先于javascript加载),比如#FF0000或者rgb(255, 0, 0),而这验证了我上面的说法。而在opera中却无法读出style的属性。代码如下:

html 文件内容
<!DOCTYPE html>
<head><linkrel="stylesheet"href="stylesheet.css"><scriptsrc="script.js"></script>
</head>
<body><divid="element">The element</div><
/body>
</html>
stylesheet.css 文件内容
#element { color: red; }script.js文件内容
document.addEventListener('DOMContentLoaded',function(){alert(getComputedStyle(document.getElementById('element'),null).color);},
false);

2、各大javascript框架如何实现domReady事件的

早期版本的浏览器是没有DOMContentLoaded事件的那么它们怎么模拟实现类似功能呢?先来说说原理

(1)、如果是webkit引擎则轮询document的readyState属性,当值为loaded或者complete时则触发DOMContentLoaded事件,对webkit525之后版本直接可以注册DOMContentLoaded事件

if(Browser.Engine.webkit){  timer = window.setInterval(function(){if(/loaded|complete/.test(document.readyState))  fireContentLoadedEvent();},0);
}

(2)、IE处理方式有多种

a、在页面临时插入一个script元素,并设置defer属性,最后把该脚本加载完成视作DOMContentLoaded事件来触发。这样做有一个问题是,如果插入脚本的页面包含iframe的话,会等到iframe加载完才触发,其实这与onload是无异的。即这个方法不准确。

b、通过setTiemout来不断的调用documentElement的doScroll方法,直到调用成功则出触发DOMContentLoaded。这样做的原理是在IE下,DOM的某些方法只有在DOM解析完成后才可以调用,doScroll就是这样一个方法,反过来当能调用doScroll的时候即是DOM解析完成之时,与prototype中的document.write相比,该方案可以解决页面有iframe时失效的问题

c、首先注册document的onreadystatechange事件,但经测试后该方法与window.onload相当,效果不大。下面是jquery做的兼容性处理代码。

document.attachEvent("onreadystatechange",function(){if( document.readyState ==="complete"){  document.detachEvent("onreadystatechange", arguments.callee );  jQuery.ready();}
});

接下来具体看一看几大前端框架是如何综合运用这几个方法的。

jQuery.ready.promise = function( obj ) {//定义一个状态机if ( !readyList ) {//保证页面只创建一个延迟对象,多次使用$.ready() 则直接使用延迟对象done方法加入回调队列readyList = jQuery.Deferred();//异步延迟对象// readyRE = /complete|loaded|interactive/,// 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:15if ( document.readyState === "complete" ) {//

这个属性是只读的,传回值有以下的可能:

//0-UNINITIALIZED:XML 对象被产生,但没有任何文件被加载。 
          //1-LOADING:加载程序进行中,但文件尚未开始解析。 
          //2-LOADED:部分的文件已经加载且进行解析,但对象模型尚未生效。 
          //3-INTERACTIVE:仅对已加载的部分文件有效,在此情况下,对象模型是有效但只读的。 
          //4-COMPLETED:文件已完全加载,代表加载成功

// Handle it asynchronously to allow scripts the opportunity to delay ready
            setTimeout( jQuery.ready );// Standards-based browsers support DOMContentLoaded} else if ( document.addEventListener ) {//符合W3C标准的浏览器// Use the handy event callbackdocument.addEventListener( "DOMContentLoaded", completed, false );// A fallback to window.onload, that will always workwindow.addEventListener( "load", completed, false );//还是给load事件注册了事件,以防不测,做为回滚用// If IE event model is used} else {// Ensure firing before onload, maybe late but safe also for iframesdocument.attachEvent( "onreadystatechange", completed );// A fallback to window.onload, that will always workwindow.attachEvent( "onload", completed );// If IE and not a frame// continually check to see if the document is readyvar top = false;try {//判断是否为iframe,如果不是的话采用不断的轮询scorll的方法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();//实际:readyList.resolveWith( document, [ jQuery ] );
                    }})();}}}return readyList.promise( obj );
};

再贴上几段其他框架的代码,大同小异,就不具体分析了

prototype

(function(GLOBAL) {/* Support for the DOMContentLoaded event is based on work by Dan Webb,Matthias Miller, Dean Edwards, John Resig, and Diego Perini. */var TIMER;function fireContentLoadedEvent() {if (document.loaded) return;if (TIMER) window.clearTimeout(TIMER);document.loaded = true;document.fire('dom:loaded');}function checkReadyState() {if (document.readyState === 'complete') {document.detachEvent('onreadystatechange', checkReadyState);fireContentLoadedEvent();}}function pollDoScroll() {try {document.documentElement.doScroll('left');} catch (e) {TIMER = pollDoScroll.defer();return;}fireContentLoadedEvent();}if (document.readyState === 'complete') {// We must have been loaded asynchronously, because the DOMContentLoaded// event has already fired. We can just fire `dom:loaded` and be done// with it.
    fireContentLoadedEvent();return;}if (document.addEventListener) {// All browsers that support DOM L2 Events support DOMContentLoaded,// including IE 9.document.addEventListener('DOMContentLoaded', fireContentLoadedEvent, false);} else {document.attachEvent('onreadystatechange', checkReadyState);if (window == top) TIMER = pollDoScroll.defer();}// Worst-case fallback.Event.observe(window, 'load', fireContentLoadedEvent);
})(this);

mootools

(function(GLOBAL) {/* Support for the DOMContentLoaded event is based on work by Dan Webb,Matthias Miller, Dean Edwards, John Resig, and Diego Perini. */var TIMER;function fireContentLoadedEvent() {if (document.loaded) return;if (TIMER) window.clearTimeout(TIMER);document.loaded = true;document.fire('dom:loaded');}function checkReadyState() {if (document.readyState === 'complete') {document.detachEvent('onreadystatechange', checkReadyState);fireContentLoadedEvent();}}function pollDoScroll() {try {document.documentElement.doScroll('left');} catch (e) {TIMER = pollDoScroll.defer();return;}fireContentLoadedEvent();}if (document.readyState === 'complete') {// We must have been loaded asynchronously, because the DOMContentLoaded// event has already fired. We can just fire `dom:loaded` and be done// with it.
    fireContentLoadedEvent();return;}if (document.addEventListener) {// All browsers that support DOM L2 Events support DOMContentLoaded,// including IE 9.document.addEventListener('DOMContentLoaded', fireContentLoadedEvent, false);} else {document.attachEvent('onreadystatechange', checkReadyState);if (window == top) TIMER = pollDoScroll.defer();}// Worst-case fallback.Event.observe(window, 'load', fireContentLoadedEvent);
})(this);

纸上学来终觉浅,绝知此事要躬行。自己写一段。

(function(window,undefined){hobo = {}var readyList = [],_isReady =false;function readyFn(){console.log(event.type)if ( document.addEventListener || event.type === "load" || document.readyState === "complete" ) {detach();_isReady =true;fireReady();
        
}}
    
  function fireReady(){
      for (var i = 0,fn; fn = readyList[i++];) {fn();};
      readyList = null;
      fireReady = function(){}//惰性函数,防止IE9二次调用
  }
function detach() {if ( document.addEventListener ) {document.removeEventListener( "DOMContentLoaded", readyFn, false );window.removeEventListener( "load", readyFn, false );} else {document.detachEvent( "onreadystatechange", readyFn );window.detachEvent( "onload", readyFn );}}hobo.ready = function(fn){if(readyList){readyList.push(fn)}if(readyList.length>1){return;} if(document.readyState === 'complete'){setTimeout(readyFn);}else if (document.addEventListener) {//符合W3C 则监听 DOMContentLoaded和load事件console.log('addEventListener')document.addEventListener('DOMContentLoaded',readyFn,false);document.addEventListener('DOMContentLoaded',readyFn,false);}else{//针对IEconsole.log('attachEvent')document.attachEvent('onreadystatechange',readyFn);document.attachEvent('onload',readyFn);}//针对IE并且非framevar top = false;try{top = window.frameElement===null&&document.documentElement}catch(e){}if(top&&top.doScroll){(function doScrollCheck(){if (!_isReady) {try {//每隔50秒轮询 检测是否支持doScroll()方法top.doScroll("left");} catch(e) {return setTimeout( doScrollCheck, 50 );}};})}}window.hobo =hobo }(window,void 0))//使用 hobo.ready(function(){console.log(11111); }) hobo.ready(function(){console.log(22222); })

 

转载于:https://www.cnblogs.com/hoboStage/p/5089912.html

相关文章:

php-fpm – 配置详解

http://duyongguang.blogbus.com/logs/156375484.html php5.3自带php-fpm /usr/local/php/etc/php-fpm.conf pid run/php-fpm.pidpid设置&#xff0c;默认在安装目录中的var/run/php-fpm.pid&#xff0c;建议开启 error_log log/php-fpm.log错误日志&#xff0c;默认在安装目…

真香!Vision Transformer 快速实现 Mnist 识别

作者 | 李秋键出品 | AI科技大本营&#xff08;ID:rgznai100&#xff09;引言&#xff1a;基于深度学习的方法在计算机视觉领域中最典型的应用就是卷积神经网络CNN。CNN中的数据表示方式是分层的&#xff0c;高层特征表示依赖于底层特征&#xff0c;由浅入深抽象地提取高级特征…

(二十一)数组的初始化

class Demo3 {public static void main(String[] args) {//数组的初始化int[] a new int[] {12,13,14,15};int[] b {12,13,14,15};//数组的便利for(int i 0;i<4;i) {System.out.println(a[i]);}for(int i 0;i<b.length;i) {System.out.println(b[i]);}} }转载于:http…

深入探讨PHP中的内存管理问题

一、 内存在PHP中&#xff0c;填充一个字符串变量相当简单&#xff0c;这只需要一个语句"&#xff1c;?php $str hello world ; ?&#xff1e;"即可&#xff0c;并且该字符串能够被自由地修改、拷贝和移动。而在C语言中&#xff0c;尽管你能够编写例如"char …

介绍一个效率爆表的数据采集框架

作者 | 俊欣来源丨关于数据分析与可视化今天我们来聊一下如何用协程来进行数据的抓取&#xff0c;协程又称为是微线程&#xff0c;也被称为是用户级线程&#xff0c;在单线程的情况下完成多任务&#xff0c;多个任务按照一定顺序交替执行。那么aiohttp模块在Python中作为异步的…

最多显示6行并且最多显示三条文本

为什么80%的码农都做不了架构师&#xff1f;>>> private void setCommentContent(ViewHolder vh, String feedId, int commentNum, ArrayList<CommentItem> comment_lists){if(commentNum < 0 || comment_lists null || comment_lists.isEmpty()){for(in…

【刷算法】LeetCode- 两数之和

题目描述 给定一个整数数组和一个目标值&#xff0c;找出数组中和为目标值的两个数。 你可以假设每个输入只对应一种答案&#xff0c;且同样的元素不能被重复利用。 示例: 给定 nums [2, 7, 11, 15], target 9因为 nums[0] nums[1] 2 7 9 所以返回 [0, 1] 复制代码分析 第…

栈区和堆区内存分配区别

一直以来总是对这个问题的认识比较朦胧&#xff0c;我相信很多朋友也是这样的&#xff0c;总是听到内存一会在栈上分配&#xff0c;一会又在堆上分配&#xff0c;那么它们之间到底是怎么的区别呢&#xff1f;为了说明这个问题&#xff0c;我们先来看一下内存内部的组织情况&…

高精度进制转换

高精度进制转换&#xff1a; 对于普通的不是非常大的数的相互转换&#xff0c;我们一般採用不断模取余的方法&#xff0c;比如&#xff1a;将10进制数m转换成n进制数&#xff0c;则对m模n取余就可以。可是&#xff0c;假设是一个有几百、几千或者很多其它位的大数呢&#xff1f…

远程办公,你希望在家工作几天?

受疫情影响&#xff0c;员工的工作方式不得不发生改变。在过去短短的几个月内&#xff0c;远程办公从偶然一次变成了常态化。随着疫情的反复&#xff0c;远程办公再次成为了许多企业的选择。3月份携程正式启动了“32”混合办公模式&#xff0c;即每周有1-2天&#xff0c;员工可…

python爬虫日志(9)爬取代理

2019独角兽企业重金招聘Python工程师标准>>> 话不多说&#xff0c;直接上代码&#xff0c;很简单&#xff0c;很容易看懂 import requests from bs4 import BeautifulSoup import randomdef get_ip_list():print("正在获取代理列表...")ip_url http://ww…

使php支持mbstring库以及使用

1.执行yum install php-mbstring2. 修改php.ini (这一步非常重要, 部分lxadmin版本无法自动修改)echo ‘extensionmbstring.so’ >>/etc/php.ini #更具php安装目录而定3. 重启web service如果是apache: service httpd restart方法二&#xff1a;php 5.36安装目录&#xf…

仿余额宝数字跳动效果 TextCounter

1、TextCounter 效果 2、TextCounter 说明 每次打开余额宝第一件事情就去看看有多少钱&#xff0c;最炫的就是看着钱在跳动相当的舒服&#xff0c;今天放出这个效果。 温馨提示&#xff1a;支持的Android版本最低的是Android 4.0.0 IceCreamSandwich &#xff08; API等级14 &a…

年仅 16 岁的黑客少年,竟是搅乱 IT 巨头的幕后主使?

整理 | 郑丽媛出品 | CSDN近来&#xff0c;黑客组织 Lapsus$ 活跃在各大科技网站&#xff1a;窃取英伟达近 1TB 的数据、泄露三星近 190GB 的机密数据、公布微软 Bing 和 Cortana 源码…不同于大部分黑客组织&#xff0c;Lapsus$ 没有刻意隐藏自己&#xff0c;反而行事非常高调…

使用硬盘,安装双系统,Win7+CentOS

我用那个U盘装了很多次都不行&#xff0c;都是说找不到文件。最后就找了一篇博客看如何安装双系统&#xff0c;最后发现原来可以用硬盘安装的。经过5个多小时终于完成了。^-^。 1.首先是分区&#xff0c;可以使用Window7自带的磁盘管理程序进行分区。&#xff08;PS 我是用Cent…

Linux 文件系统剖析

Linux 文件系统剖析 按照分层结构讨论 Linux 文件系统 M. Tim Jones, 顾问工程师, Emulex Corp. 简介&#xff1a; 在文件系统方面&#xff0c;Linux 可以算得上操作系统中的 “瑞士军刀”。Linux 支持许多种文件系统&#xff0c;从日志型 文件系统到集群文件系统和加密文件系统…

Docker构建Nginx+Tomcat动静分离架构

随着主流Nginx WEB服务器的发展&#xff0c;现在基于Nginx的WEB服务器已广泛应用于各大互联网企业。今天我们来使用docker构建我们的LinuxNginxTomcat动静分离服务器。1) 启动docker镜像查看当前系统存在的镜像&#xff0c;我这里为CentOS6.6&#xff0c;大家可以参考我第一…

硬核!Python 四种变量的代码对象和反汇编分析

作者 | 大奎整理 | 阳哥来源丨Python数据之道在Python基础的学习过程中&#xff0c;对变量和参数的理解有助于我们从更基础层面了解Python语言的运行。在这个过程中&#xff0c;还是有不少冷门和细节的地方需要进一步熟悉。今天我们来分享Python四种变量的代码对象和反汇编分析…

Python--数据存储:pickle模块的使用讲解

在机器学习中&#xff0c;我们常常需要把训练好的模型存储起来&#xff0c;这样在进行决策时直接将模型读出&#xff0c;而不需要重新训练模型&#xff0c;这样就大大节约了时间。Python提供的pickle模块就很好地解决了这个问题&#xff0c;它可以序列化对象并保存到磁盘中&…

Linux虚拟内存和物理内存精华【美】

原文地址&#xff1a; 《Playing with Virtual Memory》 http://www.snailinaturtleneck.com/blog/2011/08/30/playing-with-virtual-memory/ 扩展阅读&#xff1a; 《Understanding Memory》 http://www.ualberta.ca/CNS/RESEARCH/LinuxClusters/mem.html 《Understanding Vir…

留不住客户?该从你的系统上找找原因了

本篇文章暨 CSDN《中国 101 计划》系列数字化转型场景之一。 《中国 101 计划——探索企业数字化发展新生态》为 CSDN 联合《新程序员》、GitCode.net 开源代码仓共同策划推出的系列活动&#xff0c;寻访一百零一个数字化转型场景&#xff0c;聚合呈现并开通评选通道&#xff0…

系统配置文件备份比较

客户的系统出各种问题&#xff0c;这次出了问题整整一天都没找出原因&#xff0c;都红脸了&#xff0c;最后发现是系统配置文件被改掉了&#xff0c;简直不能忍&#xff0c;所以写了这个脚本&#xff0c;放到定时任务里面&#xff0c;每天备份比较配置文件import difflib impor…

RPC是什么?为什么要学习RPC?

随着近几年分布式、微服务架构的火热&#xff0c;RPC在开发工作中使用的越来越多&#xff0c;也变的越来越重要。 今天我们来看RPC是什么&#xff0c;为什么要了解RPC,通过学习RPC我们能掌握什么内容&#xff1f; 什么是「RPC」 RPC 全称 Remote Procedure Call, wikipedia的部…

Lua学习笔记6:C++和Lua的相互调用

曾经一直用C写代码。话说近期刚换工作。项目组中的是cocos2dx-lua&#xff0c;各种被虐的非常慘啊有木有。新建cocos2dx-lua项目。打开class能够发现&#xff0c;事实上就是C项目啦&#xff0c;只是为什么仅仅有一类Appdelegate类呢&#xff1f;哈哈,我相信聪明的你一定猜到了&…

Redis消息通知系统的实现

Redis消息通知系统的实现Posted on 2012-02-29by 老王 http://huoding.com/2012/02/29/146最近忙着用Redis实现一个消息通知系统&#xff0c;今天大概总结了一下技术细节&#xff0c;其中演示代码如果没有特殊说明&#xff0c;使用的都是PhpRedis扩展来实现的。内存比如要推送一…

用 Python 实现答题卡识别!

作者 | 棒子胡豆来源丨CSDN博客答题卡素材图片&#xff1a;思路读入图片&#xff0c;做一些预处理工作。进行轮廓检测&#xff0c;然后找到该图片最大的轮廓&#xff0c;就是答题卡部分。进行透视变换&#xff0c;以去除除答题卡外的多余部分&#xff0c;并且可以对答题卡进行校…

Confluence 6 计划任务

管理员控制台能够允许你对 Confluence 运行的计划任务进行计划的调整&#xff0c;这些计划任务将会按照你的调整按时执行。可以按照计划执行的任务如下&#xff1a; Confluence 站点备份存储优化任务&#xff0c;清理 Confluence 的临时目录中的文件和缓存索引优化任务&#xf…

PHP共享内存段

在asp.net和java中都有共享内存&#xff0c;php除了可以使用Memcached等方式变通以外其实php也是支持共享内存的&#xff01;需要安装扩展shmop 找到php安装源文件目录# cd /usr/local/php-5.4.0/ext/shmop # /usr/local/php/bin/phpize # ./configure --with-php-config/usr/l…

马尔科夫随机场的基本概念

1、随机过程&#xff1a; 描写叙述某个空间上粒子的随机运动过程的一种方法。它是一连串随机事件动态关系的定量描写叙述。随机过程与其他数学分支&#xff0c;如微分方程、复变函数等有密切联系。是自然科学、project科学及社会科学等领域研究随机现象的重要工具。 2、马尔科夫…

从事了两年 AI 研究,我学到了什么?

作者 | Tom Silver译者 | 弯月出品 | CSDN我从事人工智能研究的工作已经有两年了&#xff0c;有朋友问我都学到了什么&#xff0c;所以我想借本文分享一些迄今为止积累的经验教训。我将在本文中分享一些常见的经历&#xff0c;还会讨论相对具体的人工智能行业技巧。希望对大家能…