JavaScript继承学习笔记【新手必看】
JavaScript作为一个面向对象语言(JS是基于对象的),可以实现继承是必不可少的,由于本身并没有类的概念,所以不会像真正的面向对象编程语言通过类实现继承,但可以通过其他方法实现继承。实现继承的方法很多,狼蚁网站SEO优化就只是其中的几种。
一. 原型链继承
function Person() { //被继承的函数叫做超类型(父类,基类) this.name='mumu'; this.age='18'; } Person.prototype.name='susu';//当属性名相需就近原则,先在实例里面查找,没找到再到原型里找 function Worker(){ //继承的函数叫做子类型(子类,派生类) this.job='student'; } Worker.prototype=new Person();//通过原型链继承,超类型实例化后的对象实例,赋值给子类的原型属性 var p2=new Worker(); console.log(p2.name); console.log(p2 instanceof Object);//ture 所有的构造函数都继承自Object
以上实现继承关键在于Worker.prototype=new Person(); 将Worker的原型成为Person的一个实例,通过原型链继承。
注意在使用原型链实现继承时,不能使用对象字面量创建原型方法,因为这样会中断关系而重写原型链。
原型链继承问题
1.出现引用共享问题,他们还是共用一个空间,子类会影响父类
function Person() { this.bodys=['eye','foot']; } function Worker(){ } Worker.prototype=new Person(); var p1=new Worker(); p1.bodys.push('hand'); var p2=new Worker(); console.log(p1.bodys); console.log(p2.bodys);
2.在创建子类型的实例时,不能像超类型的构造函数中传递参数。
那么如何解决原型链的两个问题呢?那就继续看狼蚁网站SEO优化的继承方式吧~
二. 借用构造函数继承(也叫对象冒充,伪造对象或经典继承)
function Person(name,age){ this.name=name; this.age=age; this.bodys=['eye','foot']; } Person.prototype.showName=function(){ console.log(this.name); } function Worker(name,age,job){ Person.call(this,name,age); this.job=job;//子类添加属性 } var p1=new Worker('mumu','18','学生'); p1.bodys.push('hand') ; var p2=new Worker(); console.log(p1.name); console.log(p2.bodys); console.log(p1.showName());
简单分析下以上使用借用构造函数的原理Person.call(this,name,age);这句代码调用父级构造函数,继承父级属性,使用call方法调用Person构造函数改变函数执行时候的this, 这里的this-> new出来的一个Worker对象 构造函数伪装方法把Worker传给上面的Person。
当引用类型放在构造函数里面的时候就不会被共享,所以p2不受影响。
这里借用构造函数继承方式就解决了原型链不能传递参数以及引用类型共享的问题。
小知识call()和apply()方法可以改变函数执行的作用域, 简言之就是改变函数中this指向的内容。
call()和apply()都接受两个参数第一个是在其中运行函数的作用域,另一个是传递的参数。
call和apply的区别就是参数的不同.
call中的参数必须是一个个枚举出来的.
apply中的参数必须是数组或者是arguments对象
那么问题来了为什么p1.showName()结果是错误的呢?----因为借用构造函数继承方式只能继承构造函数里的属性和方法。这里也就发现了借用构造函数的一个问题。
注意由于把方法都放在构造函数里,每次我们实例化就会分配内存空间给它造成资源的浪费,所以一般我们都是把方法放在原型里,属性放在构造函数里。
借用构造函数继承问题
因为借用构造函数只能继承构造函数里的属性和方法,在超类型的原型中定义的方法对子类而言是不可见的,所以就相当于没有了原型。结果所有的方法都只能在构造函数里定义,就没有函数复用了。
那么如何解决借用构造函数所产生的问题呢?那就要看狼蚁网站SEO优化这种继承方式了
三. 组合继承(伪经典继承)
function Person(name,age){ this.name=name; this.age=age; } Person.prototype.showName=function(){ console.log(this.name); } function Worker(name,age,job){ Person.call(this,name,age);//借用构造函数 this.job=job; } Worker.prototype=new Person();//原型链继承 var p1=new Worker('mumu','18','学生'); console.log(p1.age); p1.showName();
组合继承将原型链与借用构造函数结合。
思路通过使用原型链实现原型上的属性和方法继承,借用构造函数实现实例属性的继承
以上的例子Person.call(this,name,age);借用构造函数继承了属性
Worker.prototype=new Person();原型链继承了方法 , 避免了两者的缺点,融合了它们的优点,成为最常用的继承模式。
组合继承的问题
调用两次超类型构造函数,一次是在创建子类型原型时,另一次是在子类型的构造函数内部。
要解决这个问题就要用到寄生组合式继承方式了。
四. 原型式继承
function object(proto) { function F() {} F.prototype = proto; return new F(); } var person = { name: 'mumu', friends: ['xiaxia', 'susu'] }; var anotherPerson = object(person); anotherPerson.friends.push('wen'); var yetAnotherPerson = object(person); anotherPerson.friends.push('tian'); console.log(person.friends);//["xiaxia", "susu", "wen", "tian"] console.log(anotherPerson.__proto__)//Object {name: "mumu", friends: Array[4]}
简单分析下function object(proto)是一个临时中转函数,里面的参数proto表示将要传递进入的一个对象,F()构造函数是临时新建的对象,用来存储传递过来的对象,F.prototype = proto;将对象实例赋值给F构造函数的原型对象,返回传递过来的对象的对象实例。原型式继承还是会共享引用类型的属性。
五. 寄生式继承
//临时中转函数 function object(proto) { function F() {} F.prototype = proto; return new F(); } //寄生函数 function create(proto){ var f=object(proto); f.love=function(){ return this.name; } return f; } var person = { name: 'mumu', friends: ['xiaxia', 'susu'] }; var anotherPerson = create(person); console.log(anotherPerson.love());寄生组合式继承
六. 寄生组合式继承
function object(proto) { function F() {} F.prototype = proto; return new F(); } //寄生函数 function create(Person,Worker){ var f=object(Person.prototype);//创建对象 f.constructor=Worker;//调整原型构造指针,增强对象 Worker.prototype=f;//指定对象 } function Person(name,age){ this.name=name; this.age=age; } Person.prototype.showName=function(){ console.log(this.name); } function Worker(name,age,job){ Person.call(this,name,age); this.job=job; } create(Person,Worker);//寄生组合式继承 var p1=new Person('mumu','18','学生'); p1.showName();
这种方法也是现在实现继承方法中最完美的,也是最理想的。
以上这篇JavaScript继承学习笔记【新手必看】就是长沙网络推广分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持狼蚁SEO。
编程语言
- 如何快速学会编程 如何快速学会ug编程
- 免费学编程的app 推荐12个免费学编程的好网站
- 电脑怎么编程:电脑怎么编程网咯游戏菜单图标
- 如何写代码新手教学 如何写代码新手教学手机
- 基础编程入门教程视频 基础编程入门教程视频华
- 编程演示:编程演示浦丰投针过程
- 乐高编程加盟 乐高积木编程加盟
- 跟我学plc编程 plc编程自学入门视频教程
- ug编程成航林总 ug编程实战视频
- 孩子学编程的好处和坏处
- 初学者学编程该从哪里开始 新手学编程从哪里入
- 慢走丝编程 慢走丝编程难学吗
- 国内十强少儿编程机构 中国少儿编程机构十强有
- 成人计算机速成培训班 成人计算机速成培训班办
- 孩子学编程网上课程哪家好 儿童学编程比较好的
- 代码编程教学入门软件 代码编程教程