javascript中caller和callee详解

最近学习javascript,碰到caller和callee的问题,去网上百度了很多。搜到的内容大同小益,整理总结了一下与大家分享

caller:返回一个调用function函数函数的引用(用法:function.caller)

说明:对于函数来说,caller属性只有在函数执行时才有定义。如果函数由顶层调用,caller则为null。

0){ time-- alert(handleCaller.caller)//返回调用handleCaller函数函数引用 alert(caller.caller)//返回调用caller函数函数引用 caller() } } handleCaller()

例子分析:第一次handleCaller运行的时候,两个alert返回的都是null,alert(handleCaller.caller)返回null是因为它是由顶层调用, alert(caller.caller)返回null是因为caller的认值是null。接下去caller()函数调用,caller.caller返回的是调用它的函数(handleCaller)的引用,通过caller.caller()可以再次调用handleCaller函数。第二次handleCaller运行的时候,alert(handleCaller.caller)返回的是caller代码(其实就是caller的引用),alert(caller.caller)返回的是handleCaller代码。因为函数间的调用关系是handleCaller->caller->handleCaller。之后就不断在2个函数之间交替执行。

caller指向调用当前函数函数,但是有一点,如果是在全局作用域内(即顶层window)被调用,则返回null。 代码走起

rush:js;"> ==================== function testCaller(){ if(testCaller.caller == null){ console.log('accessed at global'); }else{ console.log('accessed at ' + testCaller.caller); } }

在全局调用

rush:vb;"> testCaller(); // accessed at global

一个函数调用

rush:js;"> function a(){ testCaller(); } a(); // accessed at function a(){testCaller();}

此时,testCaller.caller指向就是 function a

callee:返回相对应的arguments的函数引用。(多用于匿名函数递归)

说明:也许你在网上看到最多的是callee返回正在执行的函数引用。我是这么理解,每个函数都有一个自己的arguments,通常是用来存放参数的。arguments有一个callee 属性,初始值就是对应自身的函数引用。当你函数执行到该语句时,arguments是认对应的是你现在执行的函数,那么arguments.callee为当前正在执行的函数的引用。当然如果你有标记过其他函数的arguments(例子中的args),自然可以用args.callee()去再次调用那个函数

rush:js;"> function a(){ alert(arguments.callee) var args = arguments function c(){ alert(arguments.callee) args.callee() } c() } a()

例子分析:例子中的arguments.callee都是认返回当前正在执行的函数的引用(a中返回a自身函数引用,c中返回c自身函数引用),而通过用args存放a函数的arguments,在内置函数c中使用args.callee()再次调用a函数

rush:js;"> ==================== function a(x){

if(x<=1)
return x;
else
return x + a(x-1);
}
a(12) // 78

这是一个极简的递归,运行结果正常。

再看看下面的调用方法

rush:js;"> var b = a; a = null; // 将a回收 b(12); // erro : 'a' is not a function

原因也简单,b=a,b=function a(){};在b调用之前,我们用了a=null。所以在 function a 运行的时候,其中的return x + a(x-1);中的a,指向的就是null,而不是 function a。 所以就报错了,如何解决这样的问题。我们将a换一种写法

rush:js;"> function a(x){ if(x<=1) return x; else return arguments.callee(x-1); // 这句是改变的地方 }

调用

rush:js;"> var b = a; a = null; b(12); // 78

原因:虽然我们将a=null了,但是函数a中并没有用到a,而是通过arguments.callee指向当前函数。 因为arguments.callee的定义就是:返回正在执行的函数

相关文章

前言 做过web项目开发的人对layer弹层组件肯定不陌生,作为l...
前言 前端表单校验是过滤无效数据、假数据、有毒数据的第一步...
前言 图片上传是web项目常见的需求,我基于之前的博客的代码...
前言 导出Excel文件这个功能,通常都是在后端实现返回前端一...
前言 众所周知,js是单线程的,从上往下,从左往右依次执行,...
前言 项目开发中,我们可能会碰到这样的需求:select标签,禁...