关于javascript 回调函数中变量作用域的讨论


Posted in Javascript onSeptember 11, 2009

1、背景
Javascript中的回调函数,相信大家都不陌生,最明显的例子是做Ajax请求时,提供的回调函数,
实际上DOM节点的事件处理方法(onclick,ondblclick等)也是回调函数。
在使用DWR的时候,回调函数可以作为第一个或者最后一个参数出现,如:
JScript code function callBack(result){ } myDwrService.doSomething(param1,param2,callBack);//DWR的推荐方式 //或者 myDwrService.doSomething(callBack,param1,param2);

2、问题描述
最近在使用Dojo+Dwr的时候,碰到一个问题:
如果回调函数是属于某个对象(记为obj1)的方法,等到DWR执行该回调函数的时候,
上下文却不是obj1。
表现的现象就是在回调函数中访问obj1的任何属性都是undefined。
版本:Dojo1.3.1和dwr2
3、模拟问题的代码
下面的测试代码可以模拟这个问题:
JScript code

<html> 
<head> 
<script type="text/javascript"><!-- 
    var context="全局"; 
    var testObj={ 
        context:"初始", 
        callback:function (str){ 
            //回调函数 
            alert("callback:我所处的上下文中,context="+this.context+",我被回调的方式:"+str); 
        } 
    }; 
    //创建一个对象,作为测试回调函数的上下文 
    testObj.context="已设置"; 
    function testCall(){ 
        callMethod(testObj.callback); 
        callObjMethod(testObj,testObj.callback); 
    } 
    function callMethod(method){ 
        method("通过默认上下文回调"); 
    } 
    function callObjMethod(obj,method){ 
        method.call(obj,"指定显式对象上下文回调"); 
    } 
// --></script> 
</head> 
<body> <a href="javascript:void(0)" onclick="testCall()">调用测试</a> </body> 
</html>

在callObjMethod方法中,我用了两种方式回调“method"方法:
第一种方式:method("通过默认上下文回调");
没有指定上下文,我们发现回调函数内部访问context的值是全局变量的值,
这说明,执行该方法的默认上下文是全局上下文。
第二种方式:method.call(obj,"指定显式对象上下文回调");
指定obj为method执行的上下文,就能够访问到对象内部的context。

4、研究DWR
因为06年使用DOJO+DWR(1.0)的时候,已经遇到过这个问题,当时没做太多功课,直接改了dwr的源代码。
现在用dwr2,于是想先看看DWR是不是对这个问题有新的处理方式,
将dwr.jar中的engine.js拿出来,查看了有关回调的相关代码(_remoteHandleCallback和_execute),
发现对回调的处理方式似乎比1.0更简单,没办法将对象和方法一起传过去。
5、做进一步的研究
因为这次DWR在项目中的使用太广泛,而且我相信这样的需求应该是可以满足的,于是没有立刻修改源码,
首先,在Google上搜Dojo+dwr,没有查到什么结论,可能Dojo的用户不是太多。
于是又搜”javascript callback object context“,得到一篇文章专门介绍java回调函数的问题:
http://bitstructures.com/2007/11/javascript-method-callbacks
最重要的一句话:
When a function is called as a method on an object (obj.alertVal()),
"this" is bound to the object that it is called on (obj).
And when a function is called without an object (func()),
"this" is bound to the JavaScript global object (window in web browsers.)
这篇文章也提供了解决方案,就是使用Closure和匿名方法,
在javascript中,在function内部创建一个function的时候,会自动创建一个closure,
而这个closure就能记住对应的function创建时的上下文。
所以,如果这样:
JScript code var closureFunc=function(){ testObj.callback(); }
那么无论在什么地方,直接调用closureFunc()和调用testObj.callback()是等价的。
详情参见上面提到的文章:http://bitstructures.com/2007/11/javascript-method-callbacks。
6、改进模拟代码
模拟代码只,我们再增加一种回调方式:
JScript code

<html> 
<head> 
<script type="text/javascript"><!-- 
    var context="全局"; 
    var testObj={ 
    context:"初始", 
    callback:function (str){ 
        //回调函数 
        alert("callback:我所处的上下文中,context="+this.context+",我被回调的方式:"+str); 
        } 
    }; 
    //创建一个对象,作为测试回调函数的上下文 
    function testCall(){ 
        callMethod(testObj.callback); 
        callWithClosure(function(param){testObj.callback(param);}); 
        testObj.context="已设置"; 
        callObjMethod(testObj,testObj.callback); 
    } 
    function callMethod(method){ method("通过默认上下文回调"); } 
    function callWithClosure(method){ method("通过Closure保持上下文回调"); } 
    function callObjMethod(obj,method){ method.call(obj,"指定显式对象上下文回调"); } 
// --></script> 
</head> 
<body> <a href="javascript:void(0)" onclick="testCall()">调用测试</a> </body> 
</html>

测试以上代码,我们可以发现,通过Closure和通过显示指定对象得到的效果一致。
7、模拟更加真实的调用情景
但是以上代码还有一个问题,通常在真实环境中,如果回调函数是对象中方法,那么发起请求的方法也处在同一个对象,
在javascript中,this也可以代表当前对象,但不能直接用在匿名function中用,比如:
JScript code var testObj={ context:"初始", callback:function (str){//回调函数 alert("callback:我所处的上下文中,context="+this.context+",我被回调的方式:"+str); }, caller:function(){ callWithClosure(function(param){this.callback(param);}); } };//创建一个对象,作为测试回调函数的上下文
以上代码中的this指的不是testObj,而是全局上下文,
需要在closure外写一个临时变量来代表this,完整的代码如下:
JScript code
<html> 
<head> 
<script type="text/javascript"><!-- 
var context="全局"; 
var testObj={ 
    context:"初始", 
    callback:function (str){ 
        //回调函数 
        alert("callback:我所处的上下文中,context="+this.context+",我被回调的方式:"+str); 
    }, 
    caller:function(){ 
        callWithClosure(function(param){this.callback(param);}); 
        var temp=this; 
        callWithClosure(function(param){temp.callback(param);}); 
    } 
}; 
//创建一个对象,作为测试回调函数的上下文 
testObj.context="已设置"; 
function testCall(){ 
    //callMethod(testObj.callback); 
    testObj.caller(); 
    //callWithClosure(function(param){testObj.callback(param);}); 
    //callObjMethod(testObj,testObj.callback); 
} function callObjMethod(obj,method){method.call(obj,"指定显式对象上下文回调"); } 
function callMethod(method){ method("通过默认上下文回调"); } 
function callWithClosure(method){ method("通过Closure保持上下文回调"); } 
function callback(str){ alert("callback:我是定义在外部的全局函数。"); } 
// --></script> 
</head> 
<body> 
<a href="javascript:void(0)" onclick="testCall()">调用测试</a> 
</body> 
</html>

8、什么是Closure
Two one sentence summaries:
a closure is the local variables for a function - kept alive after the function has returned,
or
a closure is a stack-frame which is not deallocated when the function returns. (as if a 'stack-frame' were malloc'ed instead of being on the stack!)
Javascript 相关文章推荐
贴一个在Mozilla中常用的Javascript代码
Jan 09 Javascript
jQuery实现原理的模拟代码 -6 代码下载
Aug 16 Javascript
javascript动画对象支持加速、减速、缓入、缓出的实现代码
Sep 30 Javascript
JQuery+CSS提示框实现思路及代码(纯手工打造)
May 07 Javascript
JavaScript表单通过正则表达式验证电话号码
Mar 14 Javascript
多个$(document).ready()的执行顺序实例分析
Jul 26 Javascript
jQuery中hide()方法用法实例
Dec 24 Javascript
Javascript 制作图形验证码实例详解
Dec 22 Javascript
javascript 组合按键事件监听实现代码
Feb 21 Javascript
vuex actions传递多参数的处理方法
Sep 18 Javascript
vue全局自定义指令-元素拖拽的实现代码
Apr 14 Javascript
vue项目中常见问题及解决方案(推荐)
Oct 21 Javascript
javascript 一些用法小结
Sep 11 #Javascript
JS 日期验证正则附asp日期格式化函数
Sep 11 #Javascript
jquery 简单导航实现代码
Sep 11 #Javascript
禁止JQuery中的load方法装载IE缓存中文件的方法
Sep 11 #Javascript
一个用javascript写的select支持上下键、首字母筛选以及回车取值的功能
Sep 09 #Javascript
用Javascript 和 CSS 实现脚注(Footnote)效果
Sep 09 #Javascript
Javascript 获取滚动条位置等信息的函数
Sep 08 #Javascript
You might like
php 常用类汇总 推荐收藏
2010/05/13 PHP
基于PHP CURL获取邮箱地址的详解
2013/06/03 PHP
php+memcache实现的网站在线人数统计代码
2014/07/04 PHP
php查找字符串出现次数的方法
2014/12/01 PHP
thinkphp3.2.3版本的数据库增删改查实现代码
2016/09/22 PHP
PHP排序算法之直接插入排序(Straight Insertion Sort)实例分析
2018/04/20 PHP
PHP实现获取url地址中顶级域名的方法示例
2019/06/05 PHP
javascript深入理解js闭包
2010/07/03 Javascript
Three.js源码阅读笔记(基础的核心Core对象)
2012/12/27 Javascript
jquery ajax的success回调函数中实现按钮置灰倒计时
2013/11/19 Javascript
jquery toolbar与网页浮动工具条具体实现代码
2014/01/12 Javascript
jQuery:delegate中select()不起作用的解决方法(实例讲解)
2014/01/26 Javascript
JS执行删除前的判断代码
2014/02/18 Javascript
jquery $(document).ready()和window.onload的区别浅析
2015/02/04 Javascript
JavaScript实现单击下拉框选择直接跳转页面的方法
2015/07/02 Javascript
JS实时弹出新消息提示框并有提示音响起的实现代码
2016/04/20 Javascript
深入理解JavaScript内置函数
2016/06/03 Javascript
简单实现node.js图片上传
2016/12/18 Javascript
微信小程序 开发经验整理
2017/02/15 Javascript
vue cli安装使用less的教程详解
2019/07/12 Javascript
vue通过video.js解决m3u8视频播放格式的方法
2019/07/30 Javascript
vue动态设置路由权限的主要思路
2021/01/13 Vue.js
[40:03]DOTA2上海特级锦标赛主赛事日 - 1 败者组第一轮#1EHOME VS Archon
2016/03/02 DOTA
python中mechanize库的简单使用示例
2014/01/10 Python
Python常用算法学习基础教程
2017/04/13 Python
详解Python map函数及Python map()函数的用法
2017/11/16 Python
pandas DataFrame 根据多列的值做判断,生成新的列值实例
2018/05/18 Python
一步步教你用python的scrapy编写一个爬虫
2019/04/17 Python
keras实现调用自己训练的模型,并去掉全连接层
2020/06/09 Python
西海岸男士和男童服装:Johnnie-O
2018/03/15 全球购物
渗透攻击的测试步骤
2014/06/07 面试题
shell程序如何生命变量?shell变量是弱变量吗?
2014/11/10 面试题
幼儿教育感言
2014/02/05 职场文书
主题党日活动总结
2014/07/08 职场文书
2015年七年级班主任工作总结
2015/05/21 职场文书
python实现手机推送 代码也就10行左右
2022/04/12 Python