关于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 相关文章推荐
js 键盘记录实现(兼容FireFox和IE)
Feb 07 Javascript
JS判断、校验MAC地址的2个实例
May 05 Javascript
JavaScript判断DIV内容是否为空的方法
Jan 29 Javascript
浅谈原型对象的常用开发模式
Jul 22 Javascript
VUE Error: getaddrinfo ENOTFOUND localhost
May 03 Javascript
100行代码实现一个vue分页组功能
Nov 06 Javascript
javascript实现考勤日历功能
Nov 29 Javascript
Vue退出登录时清空缓存的实现
Nov 12 Javascript
微信小程序修改数组长度的问题的解决
Dec 17 Javascript
详解vue中在循环中使用@mouseenter 和 @mouseleave事件闪烁问题解决方法
Apr 07 Javascript
ES6新增的数组知识实例小结
May 23 Javascript
jQuery实现增删改查
Dec 22 jQuery
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 5.3.x 的strtotime() 时区设定 警告信息修复
2013/08/05 PHP
PHP简单选择排序算法实例
2015/01/26 PHP
Symfony2开发之控制器用法实例分析
2016/02/05 PHP
Yii框架防止sql注入,xss攻击与csrf攻击的方法
2016/10/18 PHP
使用swoole 定时器变更超时未支付订单状态的解决方案
2019/07/24 PHP
PHP二维数组分页2种实现方法解析
2020/07/09 PHP
超级兔子让浮动层消失的前因后果
2007/03/09 Javascript
PNG背景在不同浏览器下的应用
2009/06/22 Javascript
封装的原生javascript弹出层代码
2010/09/24 Javascript
jQuery中outerHeight()方法用法实例
2015/01/19 Javascript
js精美的幻灯片画集特效代码分享
2015/08/29 Javascript
Bootstrap每天必学之简单入门
2015/11/19 Javascript
js 取消页面可以选中文字的功能方法
2018/01/02 Javascript
three.js实现炫酷的全景3D重力感应
2018/12/30 Javascript
vue.js引入外部CSS样式和外部JS文件的方法
2019/01/06 Javascript
jQuery表单元素过滤选择器用法实例分析
2019/02/20 jQuery
js实现自定义滚动条的示例
2020/10/27 Javascript
在Python程序中实现分布式进程的教程
2015/04/28 Python
Python编程实现生成特定范围内不重复多个随机数的2种方法
2017/04/14 Python
Python之自动获取公网IP的实例讲解
2017/10/01 Python
使用Python制作自动推送微信消息提醒的备忘录功能
2018/09/06 Python
Python二叉树的镜像转换实现方法示例
2019/03/06 Python
一步步教你用python的scrapy编写一个爬虫
2019/04/17 Python
解决Python安装时报缺少DLL问题【两种解决方法】
2019/07/15 Python
python中图像通道分离与合并实例
2020/01/17 Python
Python configparser模块操作代码实例
2020/06/08 Python
Python全局变量与global关键字常见错误解决方案
2020/10/05 Python
详解css3 object-fit属性
2018/07/27 HTML / CSS
纯CSS3实现绘制各种图形实现代码详细整理
2012/12/26 HTML / CSS
党课学习思想汇报
2014/01/02 职场文书
口才训练演讲稿范文
2014/09/16 职场文书
四风批评与自我批评发言稿
2014/10/14 职场文书
黄河绝恋观后感
2015/06/08 职场文书
新学期新寄语,献给新生们!
2019/11/15 职场文书
浅谈Redis位图(Bitmap)及Redis二进制中的问题
2021/07/15 Redis
MySQL提取JSON字段数据实现查询
2022/04/22 MySQL