博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
harbor1.4.0高可用部署
阅读量:6877 次
发布时间:2019-06-26

本文共 3845 字,大约阅读时间需要 12 分钟。

一、对象冒充

其原理如下:构造函数使用 this 关键字给所有属性和方法赋值(即采用类声明的构造函数方式)。因为构造函数只是一个函数,所以可使 Parent 构造函数

成为 Children 的方法,然后调用它。Children 就会收到 Parent 的构造函数中定义的属性和方法。例如,用下面的方式定义 Parent 和 Children:

]// 父类构造函数var Parent = function(name){    this.name = name;    this.sayHi = function(){        console.log("Hi! " + this.name + ".");    }};// 子类构造函数var Children = function(name){    this.method = Parent;    this.method(name); // 实现继承的关键    delete this.method;    this.getName = function(){        console.log(this.name);    }};var p = new Parent("john");var c = new Children("joe");p.sayHi(); // 输出: Hi! john.c.sayHi(); // 输出: Hi! joe.c.getName(); // 输出: jo

 

原理:就是把 Parent 构造函数放到 Children 构造函数里面执行一次。那为什么不直接执行,非要转个弯把 Parent 赋值给 Children 的 method 属性再执行呢?

这跟 this 的指向有关,在函数内 this 是指向 window 的。当将 Parent 赋值给 Children 的 method 时, this 就指向了 Children 类的实例。

 

二、原型链继承

众所周知,JavaScript 是一门基于原型的语言,在 JavaScript 中 prototype 对象的任何属性和方法都被传递给那个类的所有实例。原型链利用这种功能来实现继承机制:

// 父类构造函数var Parent = function(){    this.name = "john";    this.sayHi = function(){        console.log("Hi! " + this.name + ".");    }};// 子类构造函数var Children = function(){};Children.prototype = new Parent(); // 实现继承的关键var p = new Parent();var c = new Children();p.sayHi(); // 输出: Hi! john.c.sayHi(); // 输出: Hi! john.

 

注意:调用 Parent 的构造函数,没有给它传递参数。这在原型链中是标准做法。要确保构造函数没有任何参数。

 

三、使用 call 或 applay 方法

这个方法是与对象冒充方法最相似的方法,因为它也是通过改变了 this 的指向而实现继承:

// 父类构造函数var Parent = function(name){    this.name = name;    this.sayHi = www.2636666.cn  function(){        console.log("Hi! " + this.name + ".");    }};// 子类构造函数var Children = function(name){    Parent.call(this, name); // 实现继承的关键    this.getName = function(){        console.log(this.name);    }};var p = new Parent(dasheng178.com"john");var c = new Children("joe");p.sayHi(); // 输出: Hi! john.c.sayHi(); www.boshenyl.cn// 输出: Hi! john.c.getName(); // 输出: joe

 

apply 方法本人就不举列了,它和 call 方法的区别在于它的第二个参数必须是数组。

 

四、混合方式

对象冒充的主要问题是必须使用构造函数方式,这不是最好的选择。不过如果使用原型链,就无法使用带参数的构造函数了。如何选择呢?答案很简单,两者都用。

在 JavaScript 中创建类的最好方式是用构造函数定义属性,用原型定义方法。这种方式同样适用于继承机制:

// 父类构造函数var Parent = function(name){    this.name = name;};Parent.prototype.sayHi = function(){    console.log("Hi! " + this.name + ".");};// 子类构造函数var Children = function(name, age){    Parent.call(this, name);www.mhylpt.com/ // 实现继承的关键    this.age = age;};Children.prototype = new Parent(); // 实现继承的关键Children.prototype.getAge = function(){    console.log(this.age);};var p = new Parent("john");var c = new Children("joe",30);p.sayHi(); // 输出: Hi! john.c.sayHi(); // 输出: Hi! joe.c.getAge(); // 输出: 30

 

五、使用Object.create 方法

Object.create 方法会使用指定的原型对象及其属性去创建一个新的对象:

// 父类构造函数var Parent = function(name){    this.name = name;};Parent.prototype.sayHi = function(){    console.log("Hi! " + this.name + ".");};// 子类构造函数var Children = function(name, age){    Parent.call(this, name); // 实现继承的关键    this.age = age;};Children.prototype = Object.create(Parent.prototype); // 实现继承的关键Children.prototype.constructor = children; // @Children.prototype.getAge =www.douniu178.com function(){    console.log(this.age);};var p = new Parent("john");var c = new Children("joe",30);p.sayHi(); // 输出: Hi! john.c.sayHi(); // 输出: Hi! joe.c.getAge(); // 输出: 30

 

@ 当执行 Children.prototype = Object.create(Parent.prototype) 这个语句后,Children 的 constructor 就被改变为 Parent ,因此需要将 Children.prototype.constructor 重

新指定为 Children 自身。

 

六、extends 关键字实现继承

这个是 ES6 的语法糖,下面看下es6实现继承的方法:

class Parent {  constructor(name, age) {    this.name = name;    this.age = age;  }}class Children extends Parent {  constructor(name, age, job) {    this.job = job; // 这里会报错    super(name, age);    this.job = job; // 正确  }}

 

上面代码中,子类的constructor方法没有调用super之前,就使用this关键字,结果报错,而放在super方法之后就是正确的。子类Children的构造函数之中的super(),代表调用父类Parent的构造函数。这是必须的,否则 JavaScript 引擎会报错。

 

注意,super虽然代表了父类Parent的构造函数,但是返回的是子类Children的实例,即super内部的this指的是Children,因此super()在这里相当于Parent.prototype.constructor.call(this)

转载地址:http://szgfl.baihongyu.com/

你可能感兴趣的文章
协议中UART的两种模式 【转】
查看>>
SharePoint 2013 Farm 安装指南——Least Privilege
查看>>
C# 温故知新 基础篇(1) C#概述
查看>>
jQuery结合lhgdialog弹出窗口,关闭时出现没有权限错误
查看>>
EXTJS学习系列提高篇:第二十八篇(转载)作者殷良胜,ext2.2打造Ext.form.ComboBox系列--分页显示...
查看>>
如何完成.Net下XML文档的读写操作
查看>>
QTP的那些事--对已经存在Excel文件修改后保存时,会弹出一个询问对话框
查看>>
UVA 11174 Stand in a Line 树dp+算
查看>>
C语言中函数strcpy ,strncpy ,strlcpy的用法【转】
查看>>
mysql join 的同时可以筛选数据
查看>>
Code First开发系列之管理并发和事务
查看>>
Spark SQL概念学习系列之为什么使用 Spark SQL?(二)
查看>>
VirtualBox-Linux系统安装增强功能
查看>>
ssh/ssh2登录
查看>>
mongodb对数组元素及内嵌文档进行增删改查操作(转)
查看>>
【python3.5】安装lxml中没有etree模块的问题解决方法
查看>>
pgpool-II的性能缺陷
查看>>
spin_lock浅析【转】
查看>>
MVC前台Post/Get异步获得数据时参数的取值问题
查看>>
8086/8088指令详解
查看>>