全面理解javascript的caller,callee,call,apply概念

来源:互联网 发布:使命召唤9剧情知乎 编辑:程序博客网 时间:2024/05/21 08:42

////  todo try

 

  全面理解javascript的caller,callee,call,apply概念[转载]2006/10/23 11:31在提到上述的概念之前,首先想说说javascript中函数的隐含参数:arguments

 

Arguments

 

该对象代表正在执行的函数和调用它的函数的参数。
 
[function.]arguments[n]
参数function :选项。当前正在执行的 Function 对象的名字。 n :选项。要传递给 Function 对象的从0开始的参数值索引。

 

说明

Arguments是进行函数调用时,除了指定的参数外,还另外创建的一个隐藏对象。Arguments是一个类似数组但不是数组的对象,说它类似数组是因为其具有数组一样的访问性质及方式,可以由arguments[n]来访问对应的单个参数的值,并拥有数组长度属性 length。还有就是arguments对象存储的是实际传递给函数的参数,而不局限于函数声明所定义的参数列表,而且不能显式创建 arguments 对象。arguments 对象只有函数开始时才可用。下边例子详细说明了这些性质:

 

[javascript] view plaincopyprint?
  1. //arguments 对象的用法。   
  2.     function ArgTest(a, b){  
  3.         var i, s = "The ArgTest function expected ";  
  4.         var numargs = arguments.length;      // 获取被传递参数的数值。  
  5.         var expargs = ArgTest.length;        // 获取期望参数的数值。  
  6.         if (expargs < 2)  
  7.            s += expargs + " argument. ";  
  8.        else  
  9.           s += expargs + " arguments. ";  
  10.        if (numargs < 2)  
  11.           s += numargs + " was passed.";  
  12.        else  
  13.           s += numargs + " were passed.";  
  14.        s += "/n/n"  
  15.        for (i =0 ; i < numargs; i++){       // 获取参数内容。  
  16.        s += "   Arg " + i + " = " + arguments[i] + "/n";  
  17.        }  
  18.        return(s);                           // 返回参数列表。  
  19.    }  

 

在此添加了一个说明arguments不是数组(Array类)的代码:

 

[javascript] view plaincopyprint?
  1. Array.prototype.selfvalue = 1;  
  2. alert(new Array().selfvalue);  
  3. function testAguments(){  
  4.      alert(arguments.selfvalue);  
  5. }  

 

运行代码你会发现第一个alert显示1,这表示数组对象拥有selfvalue属性,值为1,而当你调用函数testAguments时,你会发现显示的是“undefined”,说明了不是arguments的属性,即arguments并不是一个数组对象。

 

 

 

caller


返回一个对函数的引用,该函数调用了当前函数。

functionName.caller 

functionName 对象是所执行函数的名称。


说明


对于函数来说,caller 属性只有在函数执行时才有定义。如果函数是由顶层调用的,那么 caller 包含的就是 null 。如果在字符串上下文中使用 caller 属性,那么结果和 functionName.toString 一样,也就是说,显示的是函数的反编译文本。下面的例子说明了 caller 属性的用法:

 

[javascript] view plaincopyprint?
  1. // caller demo {   
  2. function callerDemo() {  
  3.      if (callerDemo.caller) {  
  4.          var a= callerDemo.caller.toString();  
  5.          alert(a);  
  6.      } else {  
  7.          alert("this is a top function");  
  8.      }  
  9. }  
  10. function handleCaller() {  
  11.      callerDemo();  
  12. }  
 

 

 

callee
 
返回正被执行的 Function 对象,也就是所指定的 Function 对象的正文。

[function.]arguments.callee
可选项 function 参数是当前正在执行的 Function 对象的名称。
 
说明
 
callee 属性的初始值就是正被执行的 Function 对象。

callee 属性是 arguments 对象的一个成员,它表示对函数对象本身的引用,这有利于匿名函数的递归或者保证函数的封装性,例如下边示例的递归计算1到n的自然数之和。而该属性仅当相关函数正在执行时才可用。还有需要注意的是callee拥有length属性,这个属性有时候 用于验证还是比较好的。arguments.length是实参长度,arguments.callee.length是形参长度,由此可以判断调用时形参长度是否和实参长度一致。
 
示例 

[javascript] view plaincopyprint?
  1. //callee可以打印其本身   
  2.    function calleeDemo() {  
  3.         alert(arguments.callee);  
  4.    }  
  5.    //用于验证参数   
  6.    function calleeLengthDemo(arg1, arg2) {  
  7.         if (arguments.length==arguments.callee.length) {  
  8.             window.alert("验证形参和实参长度正确!");  
  9.             return;  
  10.         } else {  
  11.             alert("实参长度:" +arguments.length);  
  12.             alert("形参长度: " +arguments.callee.length);  
  13.         }  
  14.    }  
  15.    //递归计算   
  16.    var sum = function(n){  
  17.       if (n <= 0)                          
  18.       return 1;  
  19.       else  
  20.         return n +arguments.callee(n - 1)  
  21.    }  

 

比较一般的递归函数:

 

[javascript] view plaincopyprint?
  1.   
  2.  var sum = function(n){  
  3.       if (1==n) return 1;  
  4. else return n + sum (n-1);  

 

调用时:alert(sum(100));
  其中函数内部包含了对sum自身的引用,函数名仅仅是一个变量名,在函数内部调用sum即相当于调用
  一个全局变量,不能很好的体现出是调用自身,这时使用callee会是一个比较好的方法。
 
 
  apply and call
 
      它们的作用都是将函数绑定到另外一个对象上去运行,两者仅在定义参数方式有所区别:
 
       apply(thisArg,argArray);
 
       call(thisArg[,arg1,arg2…] ]);
 
  即所有函数内部的this指针都会被赋值为thisArg,这可实现将函数作为另外一个对象的方法运行的目的
  apply的说明
 
  如果 argArray 不是一个有效的数组或者不是 arguments 对象,那么将导致一个 TypeError。
  如果没有提供 argArray 和 thisArg任何一个参数,那么 Global 对象将被用作 thisArg,
  并且无法被传递任何参数。
 
  call的说明
 
  call 方法可将一个函数的对象上下文从初始的上下文改变为由 thisArg指定的新对象。
  如果没有提供 thisArg参数,那么 Global 对象被用作 thisArg
 
  相关技巧:
 
  应用call和apply还有一个技巧在里面,就是用call和apply应用另一个函数(类)以后,当前的


  函数(类)就具备了另一个函数(类)的方法或者是属性,这也可以称之为“继承”。看下面示例:

 

[javascript] view plaincopyprint?
  1. // 继承的演示   
  2.   function base() {  
  3.        this.member = " dnnsun_Member";  
  4.        this.method = function() {  
  5.            window.alert(this.member);  
  6.        }  
  7.   }  
  8.   function extend() {  
  9.        base.call(this);  
  10.        window.alert(member);  
  11.        window.alert(this.method);  
  12.   }  

 

上面的例子可以看出,通过call之后,extend可以继承到base的方法和属性。
 
 
  顺便提一下,在javascript框架prototype里就使用apply来创建一个定义类的模式,
 
  其实现代码如下:

 

[javascript] view plaincopyprint?
  1. var Class = {  
  2.    create: function() {  
  3.      return function() {  
  4.        this.initialize.apply(this, arguments);  
  5.      }  
  6.    }  
  7. }  

 


  解析:从代码看,该对象仅包含一个方法:Create,其返回一个函数,即类。但这也同时是类的
  构造函数,其中调用initialize,而这个方法是在类创建时定义的初始化函数。通过如此途径,
  就可以实现prototype中的类创建模式
 
  示例:

 

[javascript] view plaincopyprint?
  1. var vehicle=Class.create();  
  2. vehicle.prototype={  
  3.      initialize:function(type){  
  4.          this.type=type;  
  5.      }  
  6.      showSelf:function(){  
  7.          alert("this vehicle is "this.type);  
  8.      }  
  9. }  
  10.   
  11. var moto=new vehicle("Moto");  
  12. moto.showSelf();  

 

详细的关于prototype信息
 
  在实际的编程过程中不知道有没有感觉到现有方法的不足?prototype 方法应运而生!下面,将通过实例由浅入深讲解 prototype 的具体使用方法:
 
 
  1、最简单的例子,了解 prototype:
  (1) Number.add(num):作用,数字相加
  实现方法:Number.prototype.add = function(num){return(this+num);}
  试验:alert((3).add(15)) -> 显示 18
 
 
  (2) Boolean.rev(): 作用,布尔变量取反
  实现方法:Boolean.prototype.rev = function(){return(!this);}
  试验:alert((true).rev()) -> 显示 false
 
  是不是很简单?这一节仅仅是告诉读者又这么一种方法,这种方法是这样运用的。
 
 
  2、已有方法的实现和增强,初识 prototype:
  (1) Array.push(new_element)
    作用:在数组末尾加入一个新的元素
    实现方法:
    Array.prototype.push = function(new_element){
          this[this.length]=new_element;
          return this.length;
      }
    让我们进一步来增强他,让他可以一次增加多个元素!
    实现方法:
    Array.prototype.pushPro = function() {
          var currentLength = this.length;
          for (var i = 0; i < arguments.length; i++) {
              this[currentLength + i] = arguments[i];
          }
          return this.length;
      }
    应该不难看懂吧?以此类推,你可以考虑一下如何通过增强 Array.pop 来实现删除任意位置,任意多个元素(具体代码就不再细说了)
 
  (2) String.length
    作用:这实际上是 String 类的一个属性,但是由于 JavaScript 将全角、半角均视为是一个字符,在一些实际运用中可能会造成一定的问题,现在我们通过 prototype 来弥补这部不足。
    实现方法:
    String.prototype.cnLength = function(){
          var arr=this.match(/[^/x00-/xff]/ig);
          return this.length+(arr==null?0:arr.length);
      }
    试验:alert("EaseWe空间Spaces".cnLength()) -> 显示 16
    这里用到了一些正则表达式的方法和全角字符的编码原理,由于属于另两个比较大的类别,本文不加说明,请参考相关材料。
 
 
  3、新功能的实现,深入 prototype:在实际编程中所用到的肯定不只是已有方法的增强,更多的实行的功能的要求,下面我就举两个用 prototype 解决实际问题的例子:
  (1) String.left()
    问题:用过 vb 的应该都知道left函数,从字符串左边取 n 个字符,但是不足是将全角、半角均视为是一个字符,造成在中英文混排的版面中不能截取等长的字符串
    作用:从字符串左边截取 n 个字符,并支持全角半角字符的区分
    实现方法:
    String.prototype.left = function(num,mode){
          if(!//d+/.test(num))return(this);
          var str = this.substr(0,num);
          if(!mode) return str;
          var n = str.Tlength() - str.length;
          num = num - parseInt(n/2);
          return this.substr(0,num);
      }
    试验:
       alert("EaseWe空间Spaces".left(8)) -> 显示 EaseWe空间
       alert("EaseWe空间Spaces".left(8,true)) -> 显示 EaseWe空
    本方法用到了上面所提到的String.Tlength()方法,自定义方法之间也能组合出一些不错的新方法呀!
 
  (2) Date.DayDiff()
    作用:计算出两个日期型变量的间隔时间(年、月、日、周)
    实现方法:
    Date.prototype.DayDiff = function(cDate,mode){
          try{
              cDate.getYear();
          }catch(e){
              return(0);
          }
          var base =60*60*24*1000;
          var result = Math.abs(this - cDate);
          switch(mode){
              case "y":
                  result/=base*365;
                  break;
              case "m":
                  result/=base*365/12;
                  break;
              case "w":
                  result/=base*7;
                  break;
              default:
                  result/=base;
                  break;
          }
          return(Math.floor(result));
      }
    试验:alert((new Date()).DayDiff((new Date(2002,0,1)))) -> 显示 329
       alert((new Date()).DayDiff((new Date(2002,0,1)),"m")) -> 显示 10
    当然,也可以进一步扩充,得出响应的小时、分钟,甚至是秒。
 
  (3) Number.fact()
    作用:某一数字的阶乘
    实现方法:
    Number.prototype.fact=function(){
          var num = Math.floor(this);
          if(num<0)return NaN;
          if(num==0 || num==1)
              return 1;
          else
              return (num*(num-1).fact());
      }
    试验:alert((4).fact()) -> 显示 24
    这个方法主要是说明了递归的方法在 prototype 方法中也是可行的!

 

 

原创粉丝点击