第7章 函数表达式

//不要这么做

if( condition ){
      function sayHi(){
           alert('Hello!');
      }
} else {
      function sayHi(){
           alert('Yo!');
      }
}

因为不同浏览器对上面代码理解不同,ie不管condition是否为真,都会执行alert('Yo!')

//可以这么做

var sayHi;

if( condition ){
    sayHi = function(){
          alert('Hello!');
    }
} else {
    sayHi = function(){
          alert('Yo!');
    }
}

7.1 递归

经典递归函数:

function factorial(num){
  if( num <= 1 ){
        return 1;
  } else {
    return num * factorial(num-1);
  }
}

可是,遇到下面情况,会出问题:

var another = factorial;

factorial = null;

console.log(another(2))  //出错

因为在调用another时候,factorial已经不是一个函数了。

可以使用 arguments.callee 解决问题:
注意:arguments.callee 是一个指向正在执行的函数的指针。

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

7.2 闭包

闭包:指有权访问另一个函数作用域中的变量的函数。

创建闭包的常见方式,就是在一个函数内部创建另一个函数。

函数被调用时,都会发生些什么?如何创建作用域链,作用域链有什么作用?

来看这一段代码:

function compare(value1, value2){
   if( value1 < value2 ){
       return -1;
   } else if ( value1 > value2 ) {
       return 1;
   } else {
       return 0;
   }
}

var result = compare(5, 10);

以上代码中,当调用了compare()时,会创建一个作用域链,其中,arguments、value1、value2处于作用域链的第一位,全局执行环境的变量(包含result和compare)则处于第二位。

很显然,作用域链本质上是一个指向变量对象的指针列表,它只引用但不实际包含变量对象。

再来看一个函数:

function createComparisonFunction(propertyName){

  return function(object1, object2){
          var value1 = object1[propertyName];
          var value2 = object2[propertyName];

          if( value1 < value2 ){
              return -1;
          } else if( value1 > value2 ){
              return 1;
          } else {
              return 0;
          }
  }
}

var compare = createComparisonFunction('name');
var result = compare({ name: 'jack' }, { name: 'mack'} )

注意:在另一个函数内部定义的函数会将包含函数(即外部函数)的活动对象添加到自己的作用域链中。因此,
在createComparisonFunction() 函数内部定义的匿名函数的作用域链中,实际上将会包含外部函数createComparisonFunction()的活动对象。

 其实,就是说,对于闭包,它的作用域链不仅仅是自己内部,还包括自己的外部。

接着说上面的函数,

在匿名函数从createComparisonFunction()中被返回后,它的作用域链被初始化为包含createComparisonFunction()函数的活动对象和全局变量对象。
这样,匿名函数就可以访问在createComparisonFunction()中定义的所有变量。

更为重要的是,createComparisonFunction()函数在执行完毕后,其活动对象也不会被销毁,因为匿名函数的作用域链仍然在引用这个活动对象。
换句话说,当createComparisonFunction() 函数返回后,其执行环境的作用域链会被销毁,但它的活动对象仍然会留在内存中,直到匿名函数被销毁,

createComparisonFunction() 函数的活动对象才会被销毁。

例如:

//创建函数
var compareNames = createComparisonFunction('name');

//调用函数
var result = compareNames({ name: 'jack' }, { name: 'mack'} );

//解除对匿名函数的引用(以便释放内存)
compareNames = null;

就是说,创建一个匿名函数后,除非手动设置其为null来进行销毁释放内存,否则,这个匿名函数以及其引用的外部变量依然会存在于内存中。

建议:由于闭包会携带包含它的函数的作用域,因此会比其他函数占用更多的内存。过度使用闭包可能会导致内存占用过多,因此,我们建议读者只在绝对必要时再考虑使用闭包。

7.2.1 闭包与变量

作用域链的这种配置机制引出了一个值得注意的副作用,即闭包只能取得包含函数中任何变量的最后一个值。别忘了,闭包所保存的是整个变量对象,而不是某个特殊的变量。

function create(){
  var result = [];

  for(var i = 0; i < 10; i++){
    result[i] = function() {
      return i;
    }
  }
  return result
}

console.log(create())

这个函数会返回一个数组,但是每个函数都返回10。 因为每个函数的作用域链中都保存着 create() 函数的活动对象,所以它们引用的都是同一个变量 i。

怎么样让每个函数返回自己?我们可以创建另一个匿名函数强制让闭包的行为符合预期。

function create(){
  var result = [];

  for(var i = 0; i < 10; i++){
    result[i] = function(num) {
      return function(){
        return num;
      }
    }
    console.log(result[i])
  }
  return result;
}
console.log(create())

7.2.2 关于this对象

我们知道,this对象是在运行时基于函数的执行环境绑定的:在全局函数中,this等于window,而当函数被作为某个对象的方法调用时,this等于那个对象。

重点:不过,匿名函数的执行环境具有全局性,因此其this对象通常指向window。

但是,有时候,由于编写闭包的方式不同,这一点可能不会那么明显

var name = 'the window';

var object = {
  name: 'my object',

  getNameFunc: function(){
    return function() {
            return this.name;
    }
  }
}

console.log(object.getNameFunc()()); ==> the window

如何让闭包访问object中的name呢?可以这么做:

var name = 'the window';

var object = {
  name: 'my object',

  getNameFunc: function(){
    var that = this;
    return function() {
      return that.name;
    }
  }
}

console.log(object.getNameFunc()());

7.3 模仿块级作用域

由于JS没用块级作用域,即意味着在块语句内部定义的变量,全局环境中都可以访问得到。

比如:

function outputNumbers(count){
   for( var i = 0; i < count; i++ ){
        alert(i)
   }
   alert(i) ==> 5
}

alert(i) ==> 报错

outputNumbers(5);

由于没有块级作用域,在for循环之内的i变量,被定义在了其外部函数中。而在全局环境中查找不到i是因为js有函数作用域。

虽然js没有块级作用域,但是我们可以通过匿名函数来模仿块级作用域。

function outputNumbers(count){
   (function(){
       for(var i = 0; i < count; i++){
           alert(i);
       }
   })();
   alert(i)  ==> 报错。i只能在for循环中访问得到
}

在匿名函数中定义的任何变量,都会在执行结束时被销毁。因此,i只能在循环中使用,使用后即被销毁。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,265评论 6 490
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,078评论 2 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 156,852评论 0 347
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,408评论 1 283
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,445评论 5 384
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,772评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,921评论 3 406
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,688评论 0 266
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,130评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,467评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,617评论 1 340
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,276评论 4 329
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,882评论 3 312
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,740评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,967评论 1 265
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,315评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,486评论 2 348

推荐阅读更多精彩内容

  • 1、定义函数的两种方式:1)函数声明:存在函数声明提升2)函数表达式:使用前必须先赋值;匿名函数(拉姆达函数) 2...
    94小辉阅读 289评论 0 0
  • 定义函数的方式有两种:函数声明和函数表达式。 函数声明的一个重要特征就是函数声明提升,意思是在执行代码前会先读取函...
    oWSQo阅读 661评论 0 0
  • 常规方式定义函数 定义函数有两种方式,第一种方式为常规声明方式,该方式下函数可以先使用,后声明,即"函数声明提升"...
    勤劳的悄悄阅读 253评论 0 0
  • 包管理 检查.deb包内容 dpkg-deb --info arping_2.11-1_armhf.debdpkg...
    bluexiii阅读 342评论 0 1
  • 锥子脸、带美瞳、指甲贴亮片;隆起的胸部、水蛇腰以及黑丝紧绷的长腿。这是很多外围女的标准样貌。应该说,通过科技(整容...
    羊年小少阅读 1,161评论 2 20