在学习这篇博文前,请先移步我的另外一篇博文:JS 一张图理解prototype、proto和constructor的关系,先弄清楚“原型链”,这样对于理解继承会非常有效。

注意:博文中提到的“属性”,指的是“属性+方法”,这里统称为“属性”;

一、构造函数继承

var obj = new Object();   使用构造函数new一个对象实例(所以程序员天天都在谈对象,哈哈哈)

特点:

  • 实例对象继承父类的共有属性和私有属性

来个实例加深理解:

function Animal() {
this.type = '动物';
}
Animal.prototype.getType = function(){
console.log(this.type);
}
let animal = new Animal();
console.log(animal.type); // 动物
animal.getType(); // 动物

二、原型链继承

Child.prototype = new Parent();   将父类的实例作为子类的原型

特点:

  • 子类的prototype上的所有属性将被完全覆盖,所以子类的prototype属性应该在覆盖后重新定义;
  • 子类的constructor指向父类,为了构造函数的完整性,需要重新指定子类的constructor属性,方法:Child.prototype.constructor = Child;
  • 修改子类与父类同名的属性,不会修改父类的属性;这是因为父类的属性在子类的原型链上,且这些属性相当于是子类的__prototype__或者更加上一级。(这里如果理解不了,说明没理解透 JS 一张图理解prototype、proto和constructor的关系 这篇文章)
原型继承,并不是把父类的属性和方法COPY一份给子类,而是让子类的原型和父类原型之间搭建一个链接的桥梁,以后子类(或者子类的实例),可以通过原型链的查找机制,找到父类原型上的方法,从而调取这些方法使用即可。

来个实例加深理解:

function Animal() {
this.type = '动物';
}
Animal.prototype.getType = function(){
console.log(this.type);
}
function Cat(){
this.vary = '猫';
}
Cat.prototype.getVary = function(){
console.log(this.vary);
}
Cat.prototype = new Animal();
var cat = new Cat();
// cat.getVary() // 报错:cat.getVary is not a function [原因:Cat.prototype = new Animal()的操作覆盖了原型链]
console.log(cat.constructor); // Animal 这个constructor实质调用的是Animal.prototype.constructor // 修改Cat类的constructor为Cat
Cat.prototype.constructor = Cat;
console.log(cat.constructor); // Cat cat.getType(); // 动物
// 修改Cat类prototype上的getType方法,看是否影响Animal类的getType方法
Cat.prototype.getType = function(){
console.log('我是猫科类');
}
var animal = new Animal();
animal.getType(); // 动物

三、call、apply、bind继承

在子类的构造体中,使用call、apply、bind方法,让父类方法中的this指向子类的实例,也就是改变this的上下文环境。

特点:

  • 子类构造体继承父类的私有属性(继承完成后,子类和父类是没关系的)

先来个call实现原理,很重要的哦

Function.prototype.call2 = function () {
var ary = [...arguments].slice();
if (!arguments[]) {
this(...ary);
} else {
var obj = Object(arguments[]); // 将参数变成一个对象
obj.__proto__.fn = this;
obj.fn(...ary);
delete obj.__proto__.fn;
}
};

来个实例加深理解:

function Animal() {
this.type = '动物';
}
Animal.prototype.getType = function(){
console.log(this.type);
}
function Cat(){
Animal.call(this);
this.vary = '猫';
}
Cat.prototype.getVary = function(){
console.log(this.vary);
}
var cat = new Cat();
console.log(cat.type); // 动物
// cat.getType(); // Uncaught TypeError: cat.getType is not a function
cat.type = '猫科动物';
var animal = new Animal();
console.log(animal.type); // 动物

四、寄生组合继承

var child = Object.create(obj, props);

  • obj:一个对象,应该是新创建的对象的原型。
  • props:可选。该参数对象是一组属性与值,该对象的属性名称将是新创建的对象的属性名称,值是属性描述符

我先来一段Object.create的实现方式(看懂原理很重要)

Object.create =  function (o) {
var F = function () {};
F.prototype = o;
return new F();
};

我们这里只分析参数obj的骚操作,可以看出来,Object.create是内部定义一个对象,并且让F.prototype对象赋值为引进的对象/函数 o,并return出一个对象的实例。

只要看懂了原理,我们可以参考“原型链继承”的方式去理解这种继承方法;

来个实例加深理解:

function Animal() {
this.type = '动物';
}
Animal.prototype.getType = function(){
console.log(this.type);
} var cat = Object.create(new Animal(), {
vary: {
value: '猫科动物'
}
});
console.log(cat.constructor); // Animal 这个constructor实质调用的是Animal.prototype.constructor
console.log(cat.type); // 动物
cat.getType(); // 动物
console.log(cat.vary); // '猫科动物'

JS 详解对象的继承的更多相关文章

  1. 详解Javascript的继承实现(二)

    上文<详解Javascript的继承实现>介绍了一个通用的继承库,基于该库,可以快速构建带继承关系和静态成员的javascript类,好使用也好理解,额外的好处是,如果所有类都用这种库来构 ...

  2. 【three.js详解之一】入门篇

    [three.js详解之一]入门篇   开场白 webGL可以让我们在canvas上实现3D效果.而three.js是一款webGL框架,由于其易用性被广泛应用.如果你要学习webGL,抛弃那些复杂的 ...

  3. 【three.js详解之二】渲染器篇

    [three.js详解之二]渲染器篇   本篇文章将详细讲解three.js中渲染器(renderer)的设置方法. three.js文档中渲染器的分支如下: Renderers CanvasRend ...

  4. dev-server.js详解

    转载自:https://www.cnblogs.com/ye-hcj/p/7091706.html dev-server.js详解 require('./check-versions')() var ...

  5. webpack.dev.conf.js详解

    转载自:https://www.cnblogs.com/ye-hcj/p/7087205.html webpack.dev.conf.js详解 //引入当前目录下的utils.js文件模块var ut ...

  6. JS 详解 Cookie、 LocalStorage 与 SessionStorage-转载

    记录一下这些知识,有时候用到会忘记,对原文作者表达感谢. 附上原文链接:JS 详解 Cookie. LocalStorage 与 SessionStorage 基本概念 Cookie Cookie 是 ...

  7. [js高手之路]深入浅出webpack系列2-配置文件webpack.config.js详解

    接着上文,重新在webpack文件夹下面新建一个项目文件夹demo2,然后用npm init --yes初始化项目的package.json配置文件,然后安装webpack( npm install ...

  8. [js高手之路]深入浅出webpack教程系列3-配置文件webpack.config.js详解(下)

    本文继续接着上文,继续写下webpack.config.js的其他配置用法. 一.把两个文件打包成一个,entry怎么配置? 在上文中的webpack.dev.config.js中,用数组配置entr ...

  9. [js高手之路]深入浅出webpack教程系列2-配置文件webpack.config.js详解(上)

    [js高手之路]深入浅出webpack教程系列索引目录: [js高手之路]深入浅出webpack教程系列1-安装与基本打包用法和命令参数 [js高手之路]深入浅出webpack教程系列2-配置文件we ...

随机推荐

  1. ggplot2 作图

    ggplot2 作图 ggplot2是著名的R语言作图工具包,gg为Grammar of Graphics的缩写,体现了结构化作图的思想.ggplot2根据图层来作图是非常优秀的思想,官方文档在这里 ...

  2. strtok strchr strrchr strchrnul

    NAME       strchr, strrchr, strchrnul - locate character in string SYNOPSIS       #include <strin ...

  3. all to do list

    要做的任务: 1. docker 学习 2. python docker应用 3. python 异步爬虫 4. python 词云 5. Java根据代码自动生成接口文档(Swagger)  > ...

  4. python之流程控制与运算符

    第一:流程控制 一:if条件语句 计算机之所以能做很多自动化的任务,因为它可以自己做条件判断. 单分支语句: 单分支,单个条件 age = 20 if age >= 18: print('you ...

  5. 7、js对象

    在python中我们学习了面向对象,javascript也是一门面向对象语言,在JavaScript中除了null和undefined以外其他的数据类型都被定义成了对象. 本篇导航: String对象 ...

  6. 前端工程化系列[03]-Grunt构建工具的运转机制

    在前端工程化系列[02]-Grunt构建工具的基本使用这篇文章中,已经对Grunt做了简单的介绍,此外,我们还知道了该如何来安装Grunt环境,以及使用一些常见的插件了,这篇文章主要介绍Grunt的核 ...

  7. go依赖包下载加速方法及github加速

    go依赖包下载加速方法及github加速 对于https://github.com/kubernetes/kubernetes整个仓库大小为近900M,下载起来那个伤心: 方法一:使用码云 这是码云上 ...

  8. Java通过SMS短信平台实现发短信功能

    在项目中使用过发短信的功能,但那个由于公司内部的限制很麻烦,今天在网上找到一个简单的,闲来无事就把它记录如下: 本程序是通过使用中国网建提供的SMS短信平台实现的(该平台目前为注册用户提供5条免费短信 ...

  9. QGIS Server使用记录

    目录 0. 简述 1. 下载QGIS桌面64位版本 2. 下载安装QGIS Server程序 3. 下载安装Apache服务器 4.使用及问题处理 0. 简述 关于QGIS Server相关的文档很少 ...

  10. CMD 命令2

    cd  %~dp0 切换到当前脚本所有目录 批处理常用命令总结 - 批处理命令简介 目录 echo 打开回显或关闭请求回显功能,或显示消息.如果没有任何参数,echo 命令将显示当前回显设置. ech ...