从以上代码可以看出,WebKit 在向作用域链中添加对象时,使用了 'this' 关键字,并且通过判断 'this.form' 是否存在来决定是否添加 FORM 对象到作用域链中。
其他浏览器中也有类似的实现方式,但在各浏览器中,将目标对象(即绑定了此内联事件处理函数的对象)添加到作用域链中的方式有差异,判断并决定是否在作用域链中添加 FORM 对象的方法也不相同。
2.1. 各浏览器在生成这个特殊的作用域链时添加目标对象时使用的方法不同各浏览器都会将内联事件处理函数所属的元素的 DOM 对象加入到作用域链中,但加入的方式却是不同的。
如以下代码:
<input type="button" value="hello">在所有浏览器中,都将弹出 'hello'。
再修改代码以变更 INPUT 元素的内联事件处理函数的执行上下文:
<input type="button" value="hello"> <script> var $target=document.getElementsByTagName("input")[0]; var o={ onclick:$target.onclick, value:"Hi, I'm here!" }; o.onclick(); </script>在各浏览器中运行的结果如下:
IE Chrome Hi, I'm here!Firefox Safari Opera hello
可见,各浏览器将内联事件处理函数所属的元素的 DOM 对象加入到作用域链中的方式是不同的。
在 IE Chrome 中的添加方式类似以下代码:
<input type="button" value="hello"> <script> var $target=document.getElementsByTagName("input")[0]; $target.onclick=function(){ with(document){ with(this){ alert(value); } } } </script>而在 Firefox Safari Opera 中的添加方式则类似以下代码:
<input type="button" value="hello"> <script> var $target=document.getElementsByTagName("input")[0]; $target.onclick=function(){ with(document){ with($target){ alert(value); } } } </script>由于极少需要改变内联事件处理函数的执行上下文,这个差异造成的影响并不多见。
2.2. 各浏览器在生成这个特殊的作用域链时对于在何种情况下添加 FORM 对象有不同理解各浏览器都会将内联事件处理函数所属的 FORM 对象加入到作用域链中,但如何判断该元素是否“属于”一个表单对象,各浏览器的处理方式则不相同。
如以下代码:
<form action="." method="get"> <div> <span>click</span> </div> </form> <script> document.method="document.method"; </script>在各浏览器中,点击 SPAN 元素后弹出的信息如下:
IE Safari Opera getChrome Firefox document.method
可见:
IE Safari Opera 将 FORM 对象加入到了内联事件处理函数的作用域链中,是否加入 FORM 对象看起来是由这个元素是否是一个 FORM 的子孙级元素来决定的。因此在这些浏览器中,函数内的变量 'method' 最终得到的是 FORM 的 'method' 的值。 Chrome Firefox 没有将 FORM 对象加入到内联事件处理函数的作用域链中,判断是否加入 FORM 对象是看该函数绑定的目标对象的 'form' 属性是否存在。从上文中的 WebKit 的源码中可以看到 Chrome 正是使用了 'this.form' 来判断,只有目标元素是一个 FORM 的子孙级元素并且该目标元素是一个表单元素时,'form' 属性才会存在。本例中的 SPAN 元素并不是表单元素,因此变量 'method' 最终得到的是 'document.method' 的值。如果将以上代码中的 SPAN 元素更换为 INPUT 元素或其他表单元素,则在所有浏览器中的表现将一致。
3. 由于内联事件处理函数的这种特殊的作用域链而产生问题的实例 3.1. 在元素的内联事件处理函数中访问的变量意外的与该该函数作用域链中非全局对象的其他对象的属性重名时出现的问题当一个内联事件处理函数中访问的变量意外的与该函数作用域链中非全局对象(window)的其他对象的属性重名,将导致该变量的实际值不是预期值。
假设有以下代码:
<button> click here </button> <script> function onsearch(){ alert("Click!"); } </script>作者本意为点击按钮即弹出“Click!”信息,但 WebKit 引擎浏览器的 HTMLElement 对象都有一个名为 onsearch 的事件监听器,这将导致上述代码在 Chrome Safari 中不能按照预期执行。本例中由于该监听器未定义(为 null),因此将报 “Uncaught TypeError: object is not a function” 的错误。