闭包

什么是闭包

在函数外部能够读取其他函数内部变量的函数。

通俗理解的闭包: 一个内部函数引用了外部函数的变量,外部函数形成了一个闭包

闭包的特点

优点

希望一个变量长期存储在内存中。可以读取函数内部的变量,避免设置过多全局变量造成污染

缺点

  1  常驻内存,增加内存使用量。

  2  使用不当会很容易造成内存泄露,内存中能存的东西越来越少,像是其他部分被泄露了一样.

  3 设置私有变量(内部函数调用外部函数的局部变量,此时,这个局部变量就会变成内部函数的私有变量)

闭包认识

初体验

例:在函数的外部实现函数内部变量的访问

functionfun1(){

varn=999;

functionfun2(){

console.log(n);

  }

returnfun2;//fn2 就是一个闭包函数,因为他能够访问到outer函数的作用域

}

varr=fun1();

r();

原理

function fun1(){

  function fun2(){

     console.log(999);

  }

  return fun2;      //fn2 就是一个闭包函数,因为他能够访问到outer函数的作用域

}

var r=fun1();

r();  

js中的变量,函数在执行的时候都会被加载到栈中,执行完毕在弹出,当使用闭包的时候,函数执行完成后,不会弹出,因为其他的地方还要加载其内部变量

面试题 一

使用原生js给每个li绑定onclick事件,点击后输出当前li对应的索引值

  一般方式

   function test(){

       var liObj = document.getElementsByTagName('li');

       var len = liObj.length;

       for(var i=0;i<len;i++){

           liObj[i].onclick = function(){

               console.log(i);

           }

       }

   }

  test();

闭包写法1:

   <script>

    var liObj = document.getElementsByTagName('li');

   function test(){

       var len = liObj.length;

       for(var i=0;i<len;i++){

            addClick(i);

       }

   }

   function addClick(j){

       liObj[j].onclick = function(){

               console.log(j);

           }

   }

  test();

   </script>

闭包写法2:

   <script>

   var liObj = document.getElementsByTagName('li');

   function test(){

       var len = liObj.length;

       for(var i=0;i<len;i++){

          (function(j){

           liObj[j].onclick = function(){

               console.log(j);

           }

          }(i))

       }

   }


  test();

   </script>

多学一招:自定义匿名函数

(function(){})是一个标准的函数定义,但是没有复制给任何变量。所以是没有名字的函数,叫匿名函数。没有名字就无法像普通函数那样随时随地调用了,所以在他定义完成后就马上调用他,后面的括号()是运行这个函数的意思

面试题 二

不能使用this, 利用闭包,打印每一个li的文本

<ul>

  <li>第1个li</li>

  <li>第2个li</li>

  <li>第3个li</li>

  <li>第4个li</li>

  <li>第5个li</li>

  <li>第6个li</li>

</ul>

  //不能使用this, 利用闭包,打印每一个li的文本

  var lis = document.getElementsByTagName('li');

// 常规使用this的写法

  for(var i = 0; i < lis.length; i++) {

    lis[i].onclick = function(){

     console.log(this.innerText);

      console.log(lis[i].innerText);

     }

  }

  for(var i = 0; i < lis.length; i++) {

   (function(i){

     lis[i].onclick = function(){

//      console.log(this.innerText);

       console.log(lis[i].innerText);

     }

   })(i);

总结:闭包的特点

1 闭包的作用:充当一个摄像头,函数外部可以访问函数内部的变量,减少变量的生命,避免造成污染.

2缺点:内存的占用比较大,浪费内存.

    容易造成内存的泄露.

    函数内部的私有变量就不存在了.

第二节:

1.掌握闭包的特点和原理

2.掌握闭包的应用场景

写出一个简单闭包,函数father里有一个局部变量a, 在函数外(全局状态)能调到,并每次累加

function father(){

   var a = 10;

   return function(){

       a++;

       return a;

   };

}

var son = father();

console.log(son())

console.log(son())

console.log(son())

console.log(son())

第三节:

prototype原型继承

所欲函数只要创建出来,系统都会分配一个原型对象给整个函数,通过prototype找到原型对象.

  我们创建的每个函数都有一个 prototype(原型)属性。使用原型的好处是可以让所有对象实例共享它所包含的属性和方法。

定义:

1 原型是function对象的一个属性,它定义了构造函数制造出的对象的公共祖先,通过该构造函数产生的对象,可以继承该原型的属性和方法.原型也是对象.

2 利用原型的概念和特点可以提取共有属性.

3 实例化对象可通过_ _proto__查看原型

原型与对象

概念1讲解 


      function Person(){


      }

     // Person.prototype 原型  祖先

      Person.prototype.age = 18;

      Person.prototype.say = function(){

          console.log('吃饭了');

      }

      //Person.prototype.age = 22;

     let p1 = new Person();

     let p2 = new Person();

      console.log(p1);

概念2讲解 

第一步:

  function Cart(ower,color){

       this.name="BMW";

       this.lang = '3米'

       this.price = 30000;

       this.ower = ower;

       this.color = color;

   }

   let p1 = new Cart();

   let p2 = new Cart();

   console.log(p1);

   console.log(p2);

   console.log(p1===p2);

   此时可以看出构造函数的多次创建会产生多个相同函数,造成冗余太多。

   利用原型prototype解决。回顾prototype

       console.log(Cart.prototype);

   //constructor表示当前的函数属于谁

   //__proto__  ==  [[prototype]],书面用语,表示原型指针


第二步:直接使用原型提取公共属性

  //  Cart.prototype.name = "BMW";

  //  Cart.prototype.lang = "3米";

  //  Cart.prototype.price = 30000;


   Cart.prototype={

       name:'BMW',

       lang:'3m',

       price:30000

   }

   function Cart(ower,color){


       this.ower = ower;

       this.color = color;

   }

   let p1 = new Cart();

   let p2 = new Cart();

   console.log(p1);

   console.log(p2);

概念3讲解 

原型链继承

JavaScript 中描述了原型链的概念,并将原型链作为实现继承的主要方法。其基本思想是利用原型让一个引用类型继承另一个引用类型的属性和方法。简单回顾一下构造函数、原型和实例的关系:每个构造函数都有一个原型对象,原型对象都包含一个指向构造函数的指针,而实例都包含一个指向原型对象的内部指

原型链的连接点就是 __ proto __,原型是没有的!

从近的往远的访问.

_ proto _ 是实例对象,拥有对的指向原型的属性,而构造函数是没有的.

      function Person(name){

          this.name = name;

          this.age = 18;

      }

      var zs = new Person('zs');

   // console.log(Fn.prototype.constructor);

     console.log(zs.__proto__); // {constructor: ƒ} Fn的原型

       console.log(zs.__proto__.__proto__.constructor);  //Object

       console.log(zs.__proto__.__proto__.__proto__);  //null

原型链的构造

<script>

Grand.prototype.lastName='cheng';

functionGrand(){


     }

vargrondObj=newGrand();

Father.prototype=grondObj;

functionFather(){

this.name="runzhi"

     }

varfatherObj=newFather();


Son.prototype=fatherObj;

functionSon(){

this.hobbit='smoke';

     }

varsonObj=newSon();

</script>

当Son要访问一个属性不存在时,就会找到原型的指针__proto__,然后找到原型继续找

实例属性与原型属性

原型属性的操作

查,即获取数据

删,子孙不能删除,其本身可以删除

改,除非自己增,后辈无法增

增,自己增,后辈无法增

call/apply继承

作用:都是改变this的指向,只是传参的形式不同

call方法

作用: 调用该函数,并修改函数中this的指向

语法: 函数名. call(对象,[实参]);

参数详解:

第一个参数: 要让函数中this指向谁,就写谁

后面的参数: 被调用函数要传入的实参,以逗号分隔

<script>

functionPerson(name,age){

this.name=name;

this.age=age;

console.log(this);

     }

varpersonObj=newPerson('cheng',88);

​letobj={}

Person.call(obj,'feng',66);

</script>

​例1:编写person和sudent两个函数,让student中的属性和person的一样,通过call改变调用方法

  第一步:

  function Person(name,age,sex){

        this.name = name;

        this.age = age;

        this.sex = sex;

   }

   function Student(name,age,sex,tel,grade){

       this.name = name;

        this.age = age;

        this.sex = sex;

        this.tel = tel;

        this.grade = grade;

   }

   var student = new Student();

第二步:在Student中调用call,传递this

      function Person(name,age,sex){

        this.name = name;

        this.age = age;

        this.sex = sex;

        console.log(this);

   }

   function Student(name,age,sex,tel,grade){

        Person.call(this,name,age,sex,);

        this.tel = tel;

        this.grade = grade;

   }

   var student = new Student('cheng',20,'男',18336629253,'A');

   注意:this的指向

  // student() = Student.call();

例2:企业中的模块化处理 

    function Wheel(WheelSize,style){

        this.style = style;

        this.WheelSize = WheelSize;

    }

     function Sit(c,sitColor){

         this.c = c;

         this.sitColor = sitColor;

     }

     function Model(height,width,len){

           this.height = height;

           this.width = width;

           this.len = lenl

     }

     function Car(WheelSize,style,c,sitColor,height,width,len){

             Wheel.call(this,WheelSize,style);

             Sit.call(this,c,sitColor);

             Model.call(this,height,width,len);

     }

     var car = new Car(1,23,45,66,7,8,89,56);

     公司中的模块化处理,别人把一些常用的开发好了,直接去调用

apply 方法

作用: 调用该函数,并修改函数中this的指向

语法: 函数名. apply(对象,数组);

参数详解:

第一个参数: 要让函数中this指向谁,就写谁

第二个参数: 要去传入一个数组,里面存放被调用函数需要的实参

functionfn(x,y){

console.log(this);// {a : 1}

console.log(x+y);//8

}

fn.apply({a:1}, [3,5]);

bind方法

作用: 不调用函数,克隆一个新的函数,并修改新函数中this的指向,将新的函数返回

语法: 函数名. bind(对象[,实参]);

参数详解:

第一个参数: 要让函数中this指向谁,就写谁

后面的参数: 被调用函数要传入的实参,以逗号分隔

例1:实现fn中this指向的改变

functionfn(x,y){

//调用newFn时打印结果

console.log(this);//{b:2}

console.log(x+y);//8

}

varnewFn=fn.bind({b:2},3,5);

newFn();

例2:给每一个li注册一个点击事件,点击这个li之后,每隔一秒钟,打印一下当前li中的文本

       // 1.1 获取元素

       var lis = document.getElementsByTagName('li');

       // 1.2 给每一个li注册点击事件

       for(var i = 0 ; i < lis.length; i++){

           lis[i].onclick = fn;

       }


       //每一个li的事件处理函数

       function fn(){

           // console.log(this);// --> 点击的那个li

           // 1.3 在事件处理函数中,设置一个定时器

           setInterval(function(){

               // 1.4 在定时器的回调函数中,打印当前li的文本

               console.log(this.innerText);

               // console.log(this); //-->window

           }.bind(this), 1000);

           //  setInterval(function(){

           //     // 1.4 在定时器的回调函数中,打印当前li的文本

           //     console.log(this.innerText);

           //     // console.log(this); //-->window

           // }.call(this), 1000);

       }

小结:

1  当我们需要自己调用函数,并且要修改函数中this的指向的时候, 用call / apply 2  当我们不需要自己调用函数,要浏览器帮我们调用(事件处理函数, 定时器的回调函数),并且要修改函数中this的指向,我们就可以用bind.

继承

构造函数继承

我们工作中要经常创建多个具有相同属性的对象,所以经常要写构造函数.

那么构造函数创建的出来的对象该如何实现继承呢?

例:使用call构造函数的继承

function Person(name, age){

   this.name = name;

   this.age = age;

   this.sayHello = function(){

       console.log('hello, ' + '我是' + this.name );

   }

}

function Student(name, age, score){

   this.score = score;

   Person.call(this, name, age);//借用构造函数

}

var stu = new Student('zs', 18, 100);

console.log(stu); //{score : 100, name : zs, age : 18}

stu.sayHello(); //hello, 我是zs

原型继承

function Person(name, age){

   this.name = name;

   this.age = age;

   this.sayHello = function(){

       console.log('hello, ' + '我是' + this.name );

   }

}

function Student(score){

   this.score = score;

}

Student.prototype = new Person();

var stu = new Student(100);

console.log(stu); //{score : 100}

stu.sayHello(); //hello, 我是undefined

实例化的时候没有传值,所以才会出现这种情况

我们发现,方法继承下来了,但是属性却没有继承下来

混合继承

借用构造函数 + 原型继承

functionPerson(name,age){

this.name=name;

this.age=age;

this.sayHello=function(){

console.log('hello, '+'我是'+this.name);

   }

}

functionStudent(name,age,score){

this.score=score;

Person.call(this,name,age);

}

Student.prototype=newPerson();

varstu=newStudent('zs',18,100);

console.log(stu);//{score : 100, name : zs, age : 18}

stu.sayHello();//hello, 我是zs

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

推荐阅读更多精彩内容