JavaScript与Image加载事件(onload)、加载状态(complete)


Posted in Javascript onFebruary 14, 2011

昨天用jQuery插件aeImageResize,发现它更有优势:每张图片加载完后,会马上进行等比缩放。
这归于图片对象Image的加载事件onload的功劳。
查看插件的源码,发现它也依赖图片对象的complete属性和onload事件,并且特别把IE6区分对待,到底IE6在图片加载对象上,与其它浏览器有什么不同呢?

看下文:

通过js操纵DOM很多情况下都是为了实现和当前页html元素的异步载入,我谈谈对Image对象的一些认识。
看个例子:

<input type="button" name="" value="载入图片" onclick="addImg('tt.jpg')" /> 
<script type="text/javascript"> 
<!-- 
function addImg(isrc) 
{ 
var Img = new Image(); 
Img.src = isrc; 
Img.onload = function () 
{ 
document.body.appendChild(Img); 
} 
} 
//--> 
</script>

当包含上述代码的页面打开时并不载入“tt.jpg”,当点击按钮时候才载入。当载入完成后触发onload事件显示到页面上。如果你是第一次加载 “tt.jpg" 这张图片的话,运行正常。点击按钮加载并显示一张图片,如果重复点击会怎么样呢?
IE、Opera中,除了第一次加载 图片时候显示正常,之后再点击就没有反应了,刷新也一样。难道它们只触发一次 “onload”事件?是缓存机制?
FF、Chrom中,每点击一次加载一张该图片。
稍微修改下:
<input type="button" name="" value="载入图片" onclick="addImg('tt.jpg')" /> 
<script type="text/javascript"> 
<!-- 
function addImg(isrc) 
{ 
var Img = new Image(); 
Img.onload = function () 
{ 
document.body.appendChild(Img); 
} 
Img.src = isrc; 
} 
//--> 
</script>

运行后发现,奇怪的事情发生了。
所有的浏览器都一致了,都是每点击一次加载一张图片。这又是什么原因?由此可以见 IE、Opera 执行过程中并不是只触发一次 onload 事件!
联想一下 Image 对象的一些属性看看,complete、readyState(IE专属值[uninitialized,complete]) (为防止缓存影响效果请更换图片名称!)
<input type="button" name="" value="complete" onclick='alert("complete : "+Img.complete +"\nreadyState : "+Img.readyState)' /> 
<input type="button" name="" value="载入图片" onclick="addImg('mtm.jpg')" /> 
<script type="text/javascript"> 
<!-- 
var Img; 
function addImg(isrc) 
{ 
Img = new Image(); 
//Img.src = isrc; 
Img.onload = function () 
{ 
alert("complete : "+Img.complete +"\nreadyState : "+Img.readyState) 
document.body.appendChild(Img); 
} 
Img.src = isrc; 
} 
//--> 
</script>

经过以上测试,可以看出一些不同点,对于 complete 属性来讲,IE是根据图片是否显示过来判断,就是说当加载的图片显示出来后,complete 属性的值才为 true ,否则一直是 false ,和以前是否加载过该张图片没有关系,即和缓存没有关系!但是其它浏览器表现出来的确不一样,只要以前加载过该图,浏览器有缓存,complete 就为 true ,这和IE的 readyState 属性的表现一致!
至此,可以肯定的是所有的浏览器都会缓存图片!可是上面的问题到底是什么原因导致的呢?
众所周知,从缓存里加载东西的速度是很快的,那么在
... 
Img.src = isrc; 
Img.onload = ... 
...

的过程中,难道 IE、Opera 加载的速度快到,来不及追加事件?
这回加载一张根本不存在的图片看看效果:
<input type="button" name="" value="complete" onclick='alert("complete : "+Imgttmt.complete +"\nreadyState : "+Imgttmt.readyState)' /> 
<input type="button" name="" value="载入图片" onclick="addImg('mtmttyt.jpg')" /> 
<script type="text/javascript"> 
<!-- 
var Imgttmt; 
function addImg(isrc) 
{ 
Imgttmt = new Image(); 
Imgttmt.src = isrc; 
alert("complete : "+Imgttmt.complete +"\nreadyState : "+Imgttmt.readyState) 
Imgttmt.onload = function () 
{ 
alert("impossible") 
} 
} 
//--> 
</script>

可以肯定的是所有浏览器都不触发 onload 事件。
从是否缓存或已经加载过图片的角度讲,IE、Opera表现正常,complete 始终为 false ;IE的 readyState 始终为uninitialized 。
令人疑惑的是FF,其中 Imgttmt.complete 的值一直是 true ;
更令人困惑的是 Chrom,它是在最初 new Imgttmt() 的时候 Imgttmt.complete 值为 false。而之后 Imgttmt.complete 值就一直为 true 了!
如果换一张从来没有加载过的图片,FF和Chrom 的行为就一致了,都是一开始加载时, Imgttmt.complete 值为false, 之后为 true!
测试的过程中还发现,脚本的执行顺序的确会影响到类似于 onload 等事件的追加,如果在其显示后在追加事件就没有什么实际意义了!
基于 javascript 这种解释性语言的特性,在追加事件的时候一定要注意把事件追加在触发该事件的句柄之前。
Javascript 相关文章推荐
Prototype Number对象 学习
Jul 19 Javascript
XHTML下,JS浮动代码失效的问题
Nov 12 Javascript
jQuery EasyUI 中文API Button使用实例
Apr 14 Javascript
jquery+easeing实现仿flash的载入动画
Mar 10 Javascript
详解JavaScript中setSeconds()方法的使用
Jun 11 Javascript
JS实现的新浪微博大厅文字内容滚动效果代码
Nov 05 Javascript
jQuery中ScrollTo用法示例
Sep 04 Javascript
详解为Angular.js内置$http服务添加拦截器的方法
Dec 20 Javascript
JavaScript事件委托原理与用法实例分析
Jun 07 Javascript
JavaScript创建对象方式总结【工厂模式、构造函数模式、原型模式等】
Dec 19 Javascript
JS常见面试试题总结【去重、遍历、闭包、继承等】
Aug 27 Javascript
JS sort方法基于数组对象属性值排序
Jul 10 Javascript
JQuery 选择器、过滤器介绍
Feb 14 #Javascript
AJAX分页的代码(后台asp.net)
Feb 14 #Javascript
基于jquery的图片的切换(以数字的形式)
Feb 14 #Javascript
jquery isType() 类型判断代码
Feb 14 #Javascript
jquery isEmptyObject判断是否为空对象的函数
Feb 14 #Javascript
jquery each()源代码
Feb 14 #Javascript
jquery trim() 功能源代码
Feb 14 #Javascript
You might like
PHP的简易冒泡法代码分享
2012/08/28 PHP
php通过array_merge()函数合并两个数组的方法
2015/03/18 PHP
UTF-8正则表达式如何匹配汉字
2015/08/03 PHP
PHP使用Pear发送邮件(Windows环境)
2016/01/05 PHP
PHP实现可自定义样式的分页类
2016/03/29 PHP
PHP substr()函数参数解释及用法讲解
2017/11/23 PHP
PHP设计模式之原型模式定义与用法详解
2018/04/03 PHP
使用 PHP Masked Package 屏蔽敏感数据的实现方法
2019/10/15 PHP
js textarea自动增高并隐藏滚动条
2009/12/16 Javascript
html组件不可输入(只读)同时任何组件都有效
2013/04/01 Javascript
nodejs文件操作模块FS(File System)常用函数简明总结
2014/06/05 NodeJs
莱鸟介绍javascript onclick事件
2016/01/06 Javascript
jQuery封装的屏幕居中提示信息代码
2016/06/08 Javascript
jQuery实现的placeholder效果完整实例
2016/08/02 Javascript
Bootstrap警告框(Alert)插件使用方法
2017/03/21 Javascript
微信小程序实现带刻度尺滑块功能
2017/03/29 Javascript
vue实现表格数据的增删改查
2017/07/10 Javascript
使用vue2实现带地区编号和名称的省市县三级联动效果
2018/11/05 Javascript
jquery+php后台实现省市区联动功能示例
2019/05/23 jQuery
JavaScript Tab菜单实现过程解析
2020/05/13 Javascript
js制作提示框插件
2020/12/24 Javascript
python中的全局变量用法分析
2015/06/09 Python
Python程序运行原理图文解析
2018/02/10 Python
Python+OpenCV实现车牌字符分割和识别
2018/03/31 Python
如何使用Python脚本实现文件拷贝
2019/11/20 Python
Python pandas自定义函数的使用方法示例
2019/11/20 Python
python pygame实现挡板弹球游戏
2019/11/25 Python
基于python实现检索标记敏感词并输出
2020/05/07 Python
python按顺序重命名文件并分类转移到各个文件夹中的实现代码
2020/07/21 Python
西班牙第一的网上药房:PromoFarma.com
2017/04/17 全球购物
个人工作表现评语
2014/04/30 职场文书
市场营销专业毕业生求职信
2014/07/21 职场文书
党的群众路线教育实践活动督导组工作情况汇报
2014/10/28 职场文书
小学生优秀评语
2014/12/29 职场文书
迎新生晚会主持词
2015/06/30 职场文书
Python3的进程和线程你了解吗
2022/03/16 Python