函数

  • 函数表达式不用于函数声明。函数声明要求有名字,但函数表达式不需要。没有名字的函数表达式也叫作匿名函数。

函数声明的重要特性式函数声明提升(function declaration hoisting).

sayHi();  


//函数声明 和 函数声明提升
function sayHi(){
  alert("Hi");
}

//函数表达式
var functionName = function(arg0, arg1, arg2){

}

  • 递归函数应该始终使用arguments.callee来递归的调用自身,不要使用函数名–函数名可能会变化
function factorial(num){
  if(num <= 1){
    return 1
  }else{
    return num * factorial(num - 1)
  }
}

var anotherFactorail = factorial;
factorial = null;
alert(anotherFactorail(4)) //出错

以上代码把factorial() 函数保存在一个变量anotherFactorail中,再将factorial 变量设置为null. 但在调用anotherFactorail()时,由于需要执行factorial(),而 factorial不再是函数,所以会报错。

正确的做法

function factorial(num){
  if(num <= 1){
    return 1
  }else{
    return num * arguments.callee(num - 1)
  }
}

arguments.callee 是一个指向正在执行的函数的指针,因此可以用它来实现对函数的递归调用。

闭包(closure)

请看我的这篇文章

当函数内部定义了其它函数时,就创建了闭包。闭包有权访问包含函数内部的所有变量,原理如下

  • 在后台执行环境中,闭包的作用域链包含着他自己的作用域、包含函数的作用域和全局作用域
  • 通常,函数的作用域及其所有变量都会在函数执行结束后被销毁

  • 但是,当函数返回一个闭包时,这个函数的作用域会一直在内存中保存闭包不存在为止

闭包可以在javascript 中模仿块级作用域

  • 创建并立即调用一个函数,这样既可以执行其中的代码,又不会在内存中留下对函数的引用。
  • 结果就是函数内部的所有变量都会被立即销毁--除非将某些变量赋值给了包含作用域(即外部作用域)中的变量
(function (){
  //这里是一个块级作用域(block scope)
  })();


function outoutNumbers(count){
  (function(){
    for(var i = 0; i < count; i++){
      alert(i);
    }
    })();

    alert(i); //导致错误
}


  • 闭包还可以用于在对象中创建私有变量

  • JavaScript中没有正式的私有对象属性的概念,但可以使用闭包来实现公有方法,而通过公有方法可以访问在包含作用域中定义的变量

  • 有权访问私有变量的公有方法叫做特权方法 (Privileged Method)

  • 可以使用构造函数模式,原型模式来实现自定义类型的特权方法,也可以使用模块模式,增强的模块模式来实现单例的特权方法。

function MyObejct(){

  //私有变量和函数
  var privateVariable = 10;

  function privateFunction(){
    return false;
  }

  //特权方法
  this.publicMethod = function (){
    privateVariable++;
    return privateFunction();
  }
}

具体请看javascript高级程序设计 P187

JavaScript 中的函数表达式和闭包都是极其有用的特性,可以利用它们来实现很多功能。不过,因为创建闭包必须维护额外的作用域,所以过度使用它们可能会占用大量内存。

发表评论

电子邮件地址不会被公开。 必填项已用*标注