理解JavaScript函数

时间:2007-11-14 08:41:41  来源:  作者:
8T2第一天空网络

  需要注意的是,尽管下面两种创建函数的方法是等价的:8T2第一天空网络

以下是引用片段:8T2第一天空网络
function funcName(){ 8T2第一天空网络
       //函数体 8T2第一天空网络
} 8T2第一天空网络
//等价于 8T2第一天空网络
var funcName=function(){ 8T2第一天空网络
       //函数体 8T2第一天空网络
8T2第一天空网络
  但前面一种方式创建的是有名函数,而后面是创建了一个无名函数,只是让一个变量指向了这个无名函数。在使用上仅有一点区别,就是:对于有名函数,它可以出现在调用之后再定义;而对于无名函数,它必须是在调用之前就已经定义。例如:
以下是引用片段:8T2第一天空网络
<script language="JavaScript" type="text/javascript"> 8T2第一天空网络
<!-- 8T2第一天空网络
func(); 8T2第一天空网络
var func=function(){ 8T2第一天空网络
       alert(1) 8T2第一天空网络
} 8T2第一天空网络
//--> 8T2第一天空网络
</script> 
8T2第一天空网络
  这段语句将产生func未定义的错误,而:
以下是引用片段:8T2第一天空网络
<script language="JavaScript" type="text/javascript"> 8T2第一天空网络
<!-- 8T2第一天空网络
func(); 8T2第一天空网络
function func(){ 8T2第一天空网络
      alert(1) 8T2第一天空网络
} 8T2第一天空网络
//--> 8T2第一天空网络
</script> 
8T2第一天空网络
  则能够正确执行,下面的语句也能正确执行:
以下是引用片段:8T2第一天空网络
<script language="JavaScript" type="text/javascript"> 8T2第一天空网络
<!-- 8T2第一天空网络
func(); 8T2第一天空网络
var someFunc=function func(){ 8T2第一天空网络
      alert(1) 8T2第一天空网络
} 8T2第一天空网络
//--> 8T2第一天空网络
</script> 
8T2第一天空网络
  由此可见,尽管JavaScript是一门解释型的语言,但它会在函数调用时,检查整个代码中是否存在相应的函数定义,这个函数名只有是通过function funcName()形式定义的才会有效,而不能是匿名函数。8T2第一天空网络
  函数对象和其他内部对象的关系

   除了函数对象,还有很多内部对象,比如:Object、Array、Date、RegExp、Math、Error。这些名称实际上表示一个类型,可以 通过new操作符返回一个对象。然而函数对象和其他对象不同,当用typeof得到一个函数对象的类型时,它仍然会返回字符串“function”,而 typeof一个数组对象或其他的对象时,它会返回字符串“object”。下面的代码示例了typeof不同类型的情况:8T2第一天空网络

以下是引用片段:8T2第一天空网络
alert(typeof(Function))); 8T2第一天空网络
alert(typeof(new Function())); 8T2第一天空网络
alert(typeof(Array)); 8T2第一天空网络
alert(typeof(Object)); 8T2第一天空网络
alert(typeof(new Array())); 8T2第一天空网络
alert(typeof(new Date())); 8T2第一天空网络
alert(typeof(new Object())); 
8T2第一天空网络

   运行这段代码可以发现:前面4条语句都会显示“function”,而后面3条语句则显示“object”,可见new一个function实际上是返 回一个函数。这与其他的对象有很大的不同。其他的类型Array、Object等都会通过new操作符返回一个普通对象。尽管函数本身也是一个对象,但它 与普通的对象还是有区别的,因为它同时也是对象构造器,也就是说,可以new一个函数来返回一个对象,这在前面已经介绍。所有typeof返回 “function”的对象都是函数对象。也称这样的对象为构造器(constructor),因而,所有的构造器都是对象,但不是所有的对象都是构造 器。8T2第一天空网络

  既然函数本身也是一个对象,它们的类型是function,联想到C+ +、Java等面向对象语言的类定义,可以猜测到Function类型的作用所在,那就是可以给函数对象本身定义一些方法和属性,借助于函数的 prototype对象,可以很方便地修改和扩充Function类型的定义,例如下面扩展了函数类型Function,为其增加了method1方法, 作用是弹出对话框显示"function":8T2第一天空网络

以下是引用片段:8T2第一天空网络
Function.prototype.method1=function(){ 8T2第一天空网络
      alert("function"); 8T2第一天空网络
} 8T2第一天空网络
function func1(a,b,c){ 8T2第一天空网络
      return a+b+c; 8T2第一天空网络
} 8T2第一天空网络
func1.method1(); 8T2第一天空网络
func1.method1.method1(); 

   注意最后一个语句:func1.method1.mehotd1(),它调用了method1这个函数对象的method1方法。虽然看上去有点容易混 淆,但仔细观察一下语法还是很明确的:这是一个递归的定义。因为method1本身也是一个函数,所以它同样具有函数对象的属性和方法,所有对 Function类型的方法扩充都具有这样的递归性质。8T2第一天空网络

  Function是 所有函数对象的基础,而Object则是所有对象(包括函数对象)的基础。在JavaScript中,任何一个对象都是Object的实例,因此,可以修 改Object这个类型来让所有的对象具有一些通用的属性和方法,修改Object类型是通过prototype来完成的:8T2第一天空网络

以下是引用片段:8T2第一天空网络
Object.prototype.getType=function(){ 8T2第一天空网络
       return typeof(this); 8T2第一天空网络
} 8T2第一天空网络
var array1=new Array(); 8T2第一天空网络
function func1(a,b){ 8T2第一天空网络
      return a+b; 8T2第一天空网络
} 8T2第一天空网络
alert(array1.getType()); 8T2第一天空网络
alert(func1.getType()); 

  上面的代码为所有的对象添加了getType方法,作用是返回该对象的类型。两条alert语句分别会显示“object”和“function”。8T2第一天空网络
8T2第一天空网络
      将函数作为参数传递
8T2第一天空网络

   在前面已经介绍了函数对象本质,每个函数都被表示为一个特殊的对象,可以方便的将其赋值给一个变量,再通过这个变量名进行函数调用。作为一个变量,它可 以以参数的形式传递给另一个函数,这在前面介绍JavaScript事件处理机制中已经看到过这样的用法,例如下面的程序将func1作为参数传递给 func2:8T2第一天空网络

以下是引用片段:8T2第一天空网络
function func1(theFunc){ 8T2第一天空网络
      theFunc(); 8T2第一天空网络
} 8T2第一天空网络
function func2(){ 8T2第一天空网络
      alert("ok"); 8T2第一天空网络
} 8T2第一天空网络
func1(func2); 
8T2第一天空网络
  在最后一条语句中,func2作为一个对象传递给了func1的形参theFunc,再由func1内部进行theFunc的调用。事实上,将函数作为参数传递,或者是将函数赋值给其他变量是所有事件机制的基础。8T2第一天空网络

文章评论

共有 位天空网友发表了评论 查看完整内容