转载 http://www.aspxhome.com/javascript/skills/20124/1593326.htm
本周的豆知识分享就来深入研究一下window.event对象。请先看看下边的代码片断。
|
<button id=”btn”>Click!</button>
<script type=”text/javascript”>
// <![CDATA[
document.getElementById(
'btn'
).onclick =
function
(e) {
if
(
typeof
e ==
'undefined'
) e = window.event;
var
target = (
typeof
e.target ==
'undefined'
? e.srcElement : e.target);
alert(target);
}
// ]]>
</script>
|
|
<button id=”btn”>Click!</button>
<script type=”text/javascript”>
// <![CDATA[
document.getElementById(
'btn'
).onclick =
function
(e) {
if
(
typeof
e ==
'undefined'
) e = window.event;
setTimeout(
function
() {
var
target = (
typeof
e.target ==
'undefined'
? e.srcElement : e.target);
alert(target);
}, 1000);
}
// ]]>
</script>
|
|
<button id=”btn”>Click!</button>
<script type=”text/javascript”>
// <![CDATA[
document.getElementById(
'btn'
).onclick =
function
() {
setTimeout(
function
() {
alert(window.event.srcElement);
}, 1000);
}
// ]]>
</script>
|
不辜负期望,以上代码同样产生了JS错误。然后我把alert(window.event.srcElement);改成了alert(window.event),这次终于没有错误了,而弹出的消息框里的内容是null。然后我又去翻了翻MSDN Library,看到了下边这段描述:
The event object is available only during an event—that is, you can use it in event handlers but not in other code.
也就是说,与其它浏览器在事件触发之后为每个事件创建一个单独的Event对象相对,IE的所有事件公用一个Event对象,也就是window.event。因此为了避免冲突,针对某个事件的window.event对象只在该事件的事件处理函数的执行过程中有效,一旦事件处理函数执行完了,window.event就被IE设置为null了。
到此为止,似乎问题的原因已经很明了了。但是细心的同学也许会反驳,IE其实也是为每个被触发的事件创建一个单独的Event对象,只不过每次都通过window.event来引用新生成的对象。因此我写了下边这段代码来验证这种观点。
|
<button id=”btn”>Click!</button>
<script type=”text/javascript”>
// <![CDATA[
var
lastEventObj =
null
;
document.getElementById(
'btn'
).onclick =
function
() {
if
(lastEventObj ==
null
)
lastEventObj = window.event.srcElement;
else
alert(lastEventObj === window.event.srcElement);
}
// ]]>
</script>
|
按钮在第一次被点击时,该事件的Event对象被lastEventObj变量所引用。因此就算在按钮第二次被点击时window.event被设置为新的Event对象的引用,仍然可以使用lastEventObj变量来访问到第一次点击事件时的Event对象。如果IE在每次事件被触发时都创建一个新的Event对象,那么lastEventObj === window.event.srcElement应该返回false。如果IE为所有事件公用一个Event对象,只是在每次事件触发时重新设置该对象的属性的话,lastEventObj === window.event.srcElement应该返回true。经过测试,返回值是false。
问题开始变得有些诡异了。既然每次事件被触发时的Event对象都是不同的,在本文的第二段代码里边,的确是在变量e中保存了对Event对象的引用的。即使事件处理函数执行完毕后window.event被设置为null,在1秒后自动执行的匿名函数中仍然应该可以使用变量e来访问先前的Event对象,但是为什么会产生JS错误呢?问题的原因在这时再此变得扑朔迷离了。我做了一些试验后得到了一些惊人的结论,以下我直接列举出一些事实。
1、window.event对象不是真正的JavaScript 对象。
按照ECMAScript规范,JavaScript中只应该存在有两种数据类型——值类型和对象。而一切对象都应该衍生于Object对象,因此Object.prototype中定义的方法,例如toString,应该在一切对象上都可以调用。那么我们来看看下边的代码:
|
<button id=”btn”>Click!</button>
<script type=”text/javascript”>
// <![CDATA[
document.getElementById(
'btn'
).onclick =
function
() {
alert(
typeof
window.event.toString);
}
// ]]>
</script>
|
点击按钮后,弹出的消息框里竟然显示的是undefined。由此可见,任何标准的JavaScript对象应该包含的成员却在window.event对象上消失了。既然window.event已经不是一个标准的JavaScript对象了,所以如果有什么理所当然的事情在window.event上变得不对劲了也不要感到特别惊奇。
2、实际上,IE还是为所有事件公用一个Event对象。
我们直接使用下边的代码来说明问题。
|
<button id=”btn1″>Click1!</button>
<button id=”btn2″>Click2!</button>
<script type=”text/javascript”>
// <![CDATA[
var
btn1EventObj =
null
;
document.getElementById(
'btn1'
).onclick =
function
() {
btn1EventObj = window.event;
alert(btn1EventObj.srcElement.id);
}
document.getElementById(
'btn2'
).onclick =
function
() {
alert(btn1EventObj === window.event);
alert(btn1EventObj.srcElement === window.event.srcElement);
alert(btn1EventObj.srcElement.id);
}
// ]]>
</script>
|
这次我们有了两个按钮,每个按钮都绑定了一个事件处理函数。我们先点击btn1按钮,该点击事件的Event对象的引用在btn1事件处理函数中被保存在了全局变量btn1EventObj中。然后我们试着用btn1EventObj变量来访问btn1的id属性,很好,弹出的消息框中的确显示的btn1。
接下来我们点击btn2按钮,在btn2的事件处理函数中我们做了一些测试。测试btn1EventObj === window.event时,不出所料,返回了false。看来window.event已经变成了另外一个Event对象的引用。然后我们再测试btn1EventObj.srcElement === window.event.srcElement,居然返回了true。等号左边的本来应该是按钮btn1,而右边应该是按钮btn2,现在它们居然相等了。然后我们再看看btn1EventObj.srcElement.id,结果它的值变成了btn2了。事情变得明了了,对象btn1EventObj与对象window.event之间的关系可以用下边的图来表示。
IE确实为每个事件创建了一个单独的Event对象,而window.event在事件处理函数的执行过程中也总是指向最新创建的Event对象。问题是每个Event对象的属性却共享的同一个属性值。在这个例子中,当按钮btn2被点击后,共享的属性值srcElement被更新为了按钮btn2。因此通过btn1EventObj.srcElement访问到的属性值也就被改变了。
3、被共享的Event对象属性值也只在事件处理函数的执行过程中才有效
在事件处理函数执行完毕后,并不仅仅是window.event被设置为null这么简单。可以看看如下代码:
|
<button id=”btn”>Click!</button>
<script type=”text/javascript”>
// <![CDATA[
var
eventObj =
null
;
document.getElementById(
'btn'
).onclick =
function
() {
eventObj = window.event;
setTimeout(
function
() {
alert(
typeof
eventObj.srcElement);
}, 1000);
}
// ]]>
</script>
|
|
<button id=”btn”>Click!</button>
<script type=”text/javascript”>
// <![CDATA[
var
target =
null
;
document.getElementById(
'btn'
).onclick =
function
() {
target = window.event.srcElement;
setTimeout(
function
() {
alert(target.id);
}, 1000);
}
// ]]>
</script>
|
由此可以,srcElement属性的属性值在这段代码中表现出了值类型的性质,因为事件处理函数执行完毕后,即使IE把公用的属性值设置为了未知的什么什么后,保存在变量target中的属性值并没有受到印象。至于为什么通过Event对象的属性来访问属性值时属性值表现出了对象类型的性质,这个就只有IE的开发人员知道了。
以上是本周分享的有关window.event的灵异事件录。虽然到最后还是以“只有IE开发人员才知道”这种半途而废的文字收尾了,但是至少我们可以得出一些有用的结论来帮助我们在今后写代码的过程中回避类似问题:
1、在IE中,不要在事件处理函数的执行过程以外的地方来访问Event对象及其属性。
2、如果非要访问,请在事件处理函数的执行过程中用闭包等方式把Event对象的属性的属性值保存在其它变量中。