转自:http://w3help.org/zh-cn/causes/SD9022

标准参考

关于 HTML 4.01 规范中 BODY 标记的 onload 属性说明: http://www.w3.org/TR/html401/struct/global.html#h-7.5.1

关于 HTML 4.01 规范中 onload 内在事件说明:http://www.w3.org/TR/html401/interact/scripts.html#adef-onload

关于 DOM Level2 Events 规范中 load 事件说明:http://www.w3.org/TR/DOM-Level-2-Events/events.html#Events-eventgroupings-htmlevents

问题描述

页面加载完成后会触发 onload 事件,通常下会使用 window.onload 、 document.body.onload、 HTMLIFrame.onload 方法来处理他;但是各浏览器对页面 onload 事件处理方式并不一致,这些方法可能会导致页面加载完成后无法触事件处理函数。

造成的影响

window.onload 与 document.body.onload 事件相互关系不明确。

IE6 IE7 IE8 中无法通过为属性赋值方式为 IFRAME 元素绑定 load 事件处理函数。

受影响的浏览器

IE6 IE7 IE8 Firefox  

问题分析

一、window.onload 与 document.body.onload 事件以及 BODY 标签内的内联 onload 事件,三者相互关系不明确

有关这两个事件在 W3C 指定的规范中并没有说明其区别,由于 window 对象属于 BOM 范畴没有统一规范可依,各浏览器对其在此处的实现存有差异。

在微软的 MSDN 中说明 document.body.onload 属性是 window 的 onload 事件处理程序。其它浏览器厂商并没有对此属性做更细致解释。

分析以下代码:

<button onclick="alert(document.body.onload);">查看本页 document.body.onload 事件</button>
<button onclick="alert(window.onload);">查看本页 window.onload 事件</button>
<button onclick="alert(window.onload===document.body.onload);">查看document.body.onload 事件与 window.onload 事件是否为同一事件</button>
<br />
<script>
document.body.onload = function (){ // 函数 A
document.getElementById("A").innerHTML = "document.body.onload 被触发";
}
window.onload = function (){ // 函数 B
document.getElementById("B").innerHTML = "window.onload 被触发";
}
</script>
<span id="A" style="background:gold"></span>
<span id="B" style="background:gray"></span>

代码中分别对两种 onload 属性赋值了不同的事件处理函数。其中 window.onload 先于 document.body.onload 被定义。如果他们两者使用的是同一事件处理源,则会发生 window 对象中 onload 事件定义被 body 的事件处理函数所覆盖。

实际上个浏览器运行结果如下:

  IE6 IE7 IE8 Firefox Chrome Safari Opera
实际运行 window.onload 被触发
document.body.onload 事件函数 函数 B 函数 A 函数 B
window.onload 事件函数 null 函数 B
两者事件处理函数是否一致 Flase Flase true

初步可以看出:

  • 在 IE 中 window.onload 事件函数即使定义了,也会是 null ,它的事件处理函数始终是为 document.body.onload 提供的;
  • 在 Firefox 中 document.body.onload 事件不与 window.onload 事件相同,后者事件函数不会覆盖前者;
  • 在 Chrome Safari Opera 中 document.body.onload 事件处理函数被后声明的 window.onload 事件处理函数覆盖。

为了更明确以上猜测,我们将两个事件处理函数位置对调后再来观察实际效果:

window.onload = function () // 函数 B{
document.getElementById("B").innerHTML = "window.onload 被触发";
} document.body.onload = function (){ // 函数 A
document.getElementById("A").innerHTML = "document.body.onload 被触发";
}

这次试图使用 document.body.onload 事件处理函数来覆盖 window.onload,实际运行结果如下:

  IE6 IE7 IE8 Firefox Chrome Safari Opera
实际运行 document.body.onload 被触发 window.onload .onload 被触发 document.body.onload 被触发
document.body.onload 事件函数 函数 A 函数 A 函数 A
window.onload 事件函数 null 函数 B
两者事件处理函数是否一致 Flase Flase true

可以看出:

  • 无论如何在 IE 中 window.onload 事件函数即使定义了,也会是 null ,它的事件处理函数始终是为 document.body.onload 提供的。
  • 在 Firefox 中 只有 window.onload 事件会触发页面加载完成事件,document.body.onload 只是个由用户扩展的方法,并不触发相应事件处理函数。这也说明了为什么仅在 Firefox 浏览器中,两者"事件"函数均被保留。
  • 在 Chrome Safari Opera 中页面加载完成事件由 document.body.onload 和 window.onload 两者的事件处理函数定义顺序有关,后定义的覆盖之前定义的,两者最终会使用同一事件源函数处理事件内容。

在实际应用中 onload 事件还可能被写在 BODY 标签的属性内,而不在脚本标签中定义他的处理函数。此时代码如下:

<script>
window.onload=function(){
alert("window onload");
}
</script>
<body onload="alert('overridden window onload')">

此用例中首先在脚本标签内为 window.onload 事件附加了处理函数,然后有在 BODY 标签内写入 onload 事件处理代码。如果 BODY 标签内的 onload 事件处理与 window.onload 事件处理不同,就会弹出两次对话框,并且可以根据对话框出现的顺序判断哪个 onload 事件处理在前。反之则只会弹出一个对话框,如果仅出现 "window onload" 字样提示就说明 BODY 标签内 onload 属性无效,如果出现 “overridden window onload” 提示就说明 BODY 标签内 onload 事件与 window.onload 事件为同一事件处理机制,后者覆盖了前者。

各浏览器运行结果如下:

  IE6 IE7 IE8 Firefox Chrome Safari Opera
实际运行 ”overridden window onload“
两者均代表 window onload 事件 true

可见,在所有浏览器中均将,BODY 标签内的内联 onload 事件处理等同与 window.onload 事件处理。

结合上文说明可知, Firefox 中 BODY 标签内的 onload 事件属性与脚本标记内写入的 document.body.onload 事件并不一致。

二、IE6 IE7 IE8 中无法通过为属性赋值方式为 IFRAME 元素绑定 load 事件处理函数

某些情况下可能需要为 IFRAME 标记动态添加 onload 监听事件,其实现方法有通常有三种:

  1. 直接为 HTMLIFrameElement.onload 属性为 IFRAME 标记赋值事件处理函数
  2. 使用 HTMLIFrameElement.setAttribute 方法为 IFRAME 标记赋值事件处理函数
  3. 使用事件监听方式为 IFRAME 的 onload 事件绑定处理函数

根据 DOM 规范推荐使用第三种处理方式,但是由于规范在不断修正添加中,浏览器开发时可能相应的标准规范并未出现或者并不完善,这导致了各个版本浏览对此问题的实现差异。

以下将依次分析这三种方式在各浏览器内的处理情况。

1.直接为 HTMLIFrameElement.onload 属性为 IFRAME 标记赋值事件处理函数

分析以下代码:

<body>
<div id="msgA">通过 HTMLIFrameElement.onload 属性动态加入事件处理函数的 IFRAME 没有触发 onload 事件。</div>
<div id="msgB">静态 IFRAME 没有触发 onload 事件。</div>
<br />
<script> function iframeLoadA(){
document.getElementById("msgA").innerHTML = "通过 HTMLIFrameElement.onload 属性动态加入事件处理函数的 IFRAME 已经触发 onload 事件。";
}
function iframeLoadB(){
document.getElementById("msgB").innerHTML = "静态 IFRAME 已经触发 onload 事件。";
} window.onload = function (){
var iframeB = document.getElementById("ifarmeB");
var iframeA = document.createElement('iframe');
iframeA.onload = iframeLoadA;
document.body.appendChild(iframeA);
} </script>
<iframe id="ifarmeB" onload="iframeLoadB()"></iframe>
</body>

页面中存在两个 IFRAME 标记,ifrmaeA 标记为动态生成,并且动态为其 onload 属性赋值了加载完成后的处理函数。而 iframeB 标记则是在页面中静态存在的,其 onload 属性对应的事件处理函数已经被固定写入。

运行此代码,在各个浏览器中效果如下:

  IE6 IE7 IE8 Firefox Chrome Safari Opera
HTMLIFrameElement.onload = Function 没有触发 onload 事件 触发 onload 事件
静态 onload 属性定义 触发 onload 事件

可见,在 IE 浏览器中通过 HTMLIFrameElement.onload 属性赋值方式无法响应 IFRAME 标记的 onload 事件。而其他浏览器均支持此写法。

2.使用 HTMLIFrameElement.setAttribute 方法为 IFRAME 标记赋值事件处理函数

将以上代码稍作修改,将 HTMLIFrameElement.onload 赋值语句部分修改为 HTMLIFrameElement.setAttribute方法:

window.onload = function (){
var iframeB = document.getElementById("ifarmeB");
var iframeA = document.createElement('iframe');
iframeA.setAttribute("onload","iframeLoadA()");
document.body.appendChild(iframeA);
}

此时 ifrmaeA 标记的 onload 属性以及事件处理函数被 DOM 标准方法 setAttribute 写入。

运行此代码,在各个浏览器中效果如下:

  IE6 IE7 IE8(Q) IE8(S) Firefox Chrome Safari Opera
HTMLIFrameElement.setAttribute 没有触发 onload 事件 触发 onload 事件
静态 onload 属性定义 触发 onload 事件

可见,在 IE 系列浏览器中仅有 IE8(S) 支持通过 setAttribute 方法为 IFRAME 标记写入 onload 属性以及其事件处理函数。而其他浏览器均支持此写法。

3.使用事件监听方式为 IFRAME 的 onload 事件绑定处理函数

再将以上代码稍作修改,使用 DOM 规范中推荐的事件监听绑定方法为 IFRAME 动态加入 onload 事件处理函数:

function addEvent(eventName,element,fn){
if (element.attachEvent) element.attachEvent("on"+eventName,fn);
else element.addEventListener(eventName,fn,false);
}
window.onload = function (){
var iframeB = document.getElementById("ifarmeB");
var iframeA = document.createElement('iframe');
addEvent("load",iframeA,iframeLoadA);
document.body.appendChild(iframeA);
}

代码中为了兼容 IE 私有的事件绑定方式,封装了名为 addEvent 的处理函数,他为 IE 和其他浏览器提供不同的事件绑定方法,以达到兼容效果。

运行此代码,在各个浏览器中效果如下:

  IE6 IE7 IE8(Q) IE8(S) Firefox Chrome Safari Opera
onload 事件监听绑定 触发 onload 事件
静态 onload 属性定义

此时所有浏览器中为 IFRAME 标记动态绑定的 onload 事件处理方法均生效。

综合以上三种常见情况可以得出结论,除 IE 以外的浏览器均支持对以上三种动态定义 onload 事件处理函数的方法,而所有 IE 系列浏览器对 HTMLIFrameElement.onload 属性的赋值不被支持,他们可以通过 DOM 标准方法 setAttribute 来为 HTMLIFrameElement 类型元素设置 onload 属性,但是由于 IE8 以下版本对 setAttribute 方法支持有限,导致仅 IE8(S) 下生效。最终 IE 浏览器只能通过事件监听方式为 HTMLIFrameElement 类型元素绑定 onload 事件处理函数。

解决方案

  1. 统一为 window 对象的 onload 事件绑定函数,避免在 Firefox 中产生 document.body.onload 事件理解歧义。
  2. 统一使用 DOM 规范的事件监听方法(或 IE 专有事件绑定方法)为 IFRAME 标记绑定 onload 事件处理函数。

iframe 动态onload事件处理方式的更多相关文章

  1. IFRAME动态加载触发onload事件(转)

    原文地址:http://blog.ops.cc/webtech/javascript/f5nhm.html <body> <script>var iframe = docume ...

  2. iframe动态创建及释放内存

    近期參与一个项目的开发,因为项目是基于浏览器的胖client(RIA)应用程序,页面中大量调用iframe.后期測试发现浏览器内存一直居高不下,并且打开iframe页面越多内存占用越大.在IE系列浏览 ...

  3. Selenium定位iframe动态ID

    Selenium定位iframe动态ID. 126邮箱实例 买了本虫师的书来学习selenium2自动化测试,然后写第一个实例就遇到了一些坑,好在有热心的网友提供了帮助,解决了问题 要学习seleni ...

  4. window/body/img/iframe 的onload事件

    在html页面中,只有body,img,iframe这一类标签具有onload事件. onload事件表示在当前元素载入完成后发生的事件.其中,window也有onload事件,但是跟body的是同一 ...

  5. [转]DOM0,DOM2,DOM3事件处理方式区别

    转 DOM0,DOM2,DOM3事件处理方式区别 2016年07月13日 15:00:29 judyge 阅读数:1457更多 个人分类: js与前端   引子:        文档对象模型是一种与编 ...

  6. 获取iframe子页面内容高度给iframe动态设置高度

    <!DOCTYPE html><html> <head> <meta charset="UTF-8" /> <meta nam ...

  7. Java EE开发平台随手记5——Mybatis动态代理接口方式的原生用法

    为了说明后续的Mybatis扩展,插播一篇广告,先来简要说明一下Mybatis的一种原生用法,不过先声明:下面说的只是Mybatis的其中一种用法,如需要更深入了解Mybatis,请参考官方文档,或者 ...

  8. 把多个JavaScript函数绑定到onload事件处理函数上

    为了让函数只在页面加载完毕后才得到执行,我们会把函数绑定到onload事件上: window.onload = userFunction 但如果有两个函数:firstFunction() 和 seco ...

  9. C#主要支持 5 种动态创建对象的方式

    C#主要支持 5 种动态创建对象的方式: 1. Type.InvokeMember 2. ContructorInfo.Invoke 3. Activator.CreateInstance(Type) ...

随机推荐

  1. python调用windows api

    import ctypes # 方式一 ctypes.windll.user32.MessageBoxA(None, 'message', 'title', 0) # 方式二 ctypes.WinDL ...

  2. 传智播客C++第五期培训视频教程免费下载

    C/C++的应用领域几乎无处不在,服务器,嵌入式,物联网,移动互联网,信息安全,游戏,基本上大小通吃.C/C++市场份额高达26%,也就是每四个程序员就有一个C/C++程序员.市场需求量非常大,而且工 ...

  3. python数字图像处理(1):环境安装与配置

    一提到数字图像处理编程,可能大多数人就会想到matlab,但matlab也有自身的缺点: 1.不开源,价格贵 2.软件容量大.一般3G以上,高版本甚至达5G以上. 3.只能做研究,不易转化成软件. 因 ...

  4. HoloLens开发手记 - Known issues 已知问题

    本文主要提及一份问题清单,这些问题都可能对我们开发HoloLens应用造成困扰. Visual Studio 在使用VS 2015 Update 1连接HoloLens时,可能会有些小问题.但是这些小 ...

  5. 删除 windows 下 node_modules 过深的目录

    本文同步自我的个人博客:http://www.52cik.com/2015/11/13/node-modules-del.html 说到 node 的模块,确实既好用又蛋疼.相信无数人吐槽 node_ ...

  6. Java:静态代理 and 动态代理

    代理模式是常用的设计模式,其特征是代理类与委托类具有相同的接口,在具体实现上,有静态代理和动态代理之分.代理类与委托类之间通常会存在关联关系,一个代理类的对象与一个委托类的对象关联,代理类的对象本身并 ...

  7. IDL简介与corba入门案例

    IDL接口定义语言简介   IDL用中立语言的方式进行描述,能使软件组建(不同语言编写的)间相互通信. IDL提供了一个桥来连接不同的系统. Corba 上的服务用IDL描述,将被映射为某种程序设计语 ...

  8. express的session函数

    key:这个表示session返回来的cookie的键值, 我们整理一下哈: 这个是我们没有清缓存然后刷新了一下哈,对比的结果,发现session保存的数据中,只是expires这个改变了 { &qu ...

  9. 大型网站系统架构实践(五)深入探讨web应用高可用方案

    从上篇文章到这篇文章,中间用了一段时间准备,主要是想把东西讲透,同时希望大家给与一些批评和建议,这样我才能有所进步,也希望喜欢我文章的朋友,给个赞,这样我才能更有激情,呵呵. 由于本篇要写的内容有点多 ...

  10. AngularJS开发指南16:AngularJS构建大型Web应用详解

    AngularJS是由Google创建的一种JS框架,使用它可以扩展应用程序中的HTML功能,从而在web应用程序中使用HTML声明动态内容.在该团队工作的软件工程师Brian Ford近日撰写了一篇 ...