JavaScript里的类和继承

来源:互联网 发布:克林战斗力官方数据 编辑:程序博客网 时间:2024/05/21 17:06

JavaScript与大部分客户端语言有几点明显的不同:

JS是 动态解释性语言,没有编译过程,它在程序运行过程中被逐行解释执行

JS是 弱类型语言,它的变量没有严格类型限制

JS是面向对象语言,但 没有明确的类的概念(虽然有class关键字,然而目前并没有什么卵用)

JS虽然没有类,但可以通过一些方法来模拟类以及实现类的继承。

一切皆对象,还先从对象说起。

1、对象(Object)

ECMA-262对对象的定义是:无序属性的集合,其属性可以包含基本值、对象或者函数。

直观点描述,就是由多个键值对组成的散列表。

JS创建对象的方法和其它语言大同小异:

// 通过构造函数创建

var zhangsan = new Object();

zhangsan.name = "张三";

zhangsan.age = 20;

zhangsan.sayHi = function() {

alert("Hi, I'm " + this.name);

};

// 通过对象字面量创建

var lisi = {

name: "李四",

age: 21,

sayHi: function() {

alert("Hi, I'm " + this.name);

}

};

当需要大量创建相同结构的对象时,可以使用 对象工厂(Object Factory):

// 对象工厂

function createPerson(name, age) {

return {

name: name,

age: age,

sayHi: function() {

alert("Hi, I'm " + this.name);

}

};

}

var zhangsan = createPerson("张三", 20);

var lisi = createPerson("李四", 21);

但通过这种方式创建出来的实例,不能解决类型识别问题,只知道它是一个对象,但具体什么?无法判断:

zhangsan instanceof ?

lisi.constructor = ?

这时,“类”就登场了。

2、类(Class)

2.1、构造函数模式

事实上,JS中每个函数(function)本身就是一个构造函数(constructor),就是一个类:

// 构造函数模式

function Person(name, age) {

this.name = name;

this.age = age;

this.sayHi = function() {

alert("Hi, I'm " + this.name);

};

}

var zhangsan = new Person("张三", 20);

var lisi = new Person("李四", 21);

alert(zhangsan instanceof Person); // true

alert(lisi.constructor === Person); // true

这里面其实有个问题:

alert(zhangsan.sayHi === lisi.sayHi); // false

多个实例中的同名方法并不相等,也就是说存在多个副本。而这些行为是相同的,应该指向同一个引用才对。

为了解决这个问题,JS为每个函数分配了一个 prototype(原型)属性,该属性是一个指针,指向一个对象,而这个对象的用途是包含可以由特定类型的所有实例共享的属性和方法。

2.2、原型模式

原型(Prototype):指向一个对象,作为所有实例的基引用(base reference)。

// 构造函数+原型组合模式

function Person(name, age) {

this.name = name;

this.age = age;

}

Person.prototype.sayHi = function() {

alert("Hi, I'm " + this.name);

};

var zhangsan = new Person("张三", 20);

var lisi = new Person("李四", 21);

alert(zhangsan.sayHi === lisi.sayHi); // true

在Person中,sayHi 是 原型成员(prototype),name 和 age 是 特权成员(privileged),它们都是 公共成员(public)。

注:“特权”是道格拉斯提出的名词。道格拉斯·克罗克福德(Douglas Crockford),Web界人称道爷,JSON创立者,《JavaScript语言精粹》作者,JSLint、JSMin、ADsafe开发者。

类的原型带有一个 constructor 属性,指向该类的构造函数(如果重新分配了原型指针,需要手动添加 constructor 属性);类的实例上会自动生成一个属性指向该类原型(在Chrome上可以通过“__proto__”访问到该对象,而IE上该属性则是不可见的)。

Person、Person的原型、Person的实例间的关系如下:

 

 

需要注意的是,原型成员保存引用类型值时需谨慎:

Person.prototype.friends = [];

zhangsan.friends.push("王五");

alert(lisi.friends); // ["王五"]

张三的基友莫名其妙就变成李四的基友了,所以 friends 应该添加为特权成员,而不是原型成员。

2.3、类的结构

综上所述,JS中的类的结构大致如下:

 

 

类由构造函数和原型组成

构造函数中可以声明私有成员和添加特权成员

原型中可以添加原型成员

私有成员可以被特权成员访问而对原型成员不可见

特权成员和原型成员都是公共成员

3、继承(Inherit)

在JS中继承是如何实现的呢?

3.1、拷贝继承

最简单直接的方式莫过于 属性拷贝:

// 拷贝继承

function extend(destination, source) {

for (var property in source) {

destination[property] = source[property];

}

}

extend(SubClass.prototype, SuperClass.prototype);

这种方式虽然实现了原型属性的继承,但有一个非常明显的缺陷:子类实例无法通过父类的 instanceof 验证,换句话说,子类的实例不是父类的实例。

3.2、原型继承

在 Chrome 的控制台中查看 HTMLElement 的原型,大致如下:

 

 

可以清晰看到,HTMLElement 的原型是 Element 的实例,而 Element 的原型又是 Node 的实例,从而形成了一条 原型链(Prototype-chain),JS的原生对象就是通过原型链来实现继承。

这里顺道说下解释器对实例属性的查找过程:

在特权属性中查找

特权属性中没找到,再到原型属性中查找

原型属性中没找到,再到原型的原型属性中查找

直到根原型还没找到,返回 undefined

这就说明为什么我们自定义的类明明没有声明 toString() 方法,但仍然可以访问到,因为所有对象都继承自 Object。

因此,我们也可以通过原型链来实现继承:

// 原型链继承

function User(name, age, password) {

// 继承特权成员

Person.call(this, name, age);

this.password = password;

}

// 继承原型

User.prototype = new Person();

// 修改了原型指针,需重新设置 constructor 属性

User.prototype.constructor = User;

var zhangsan = new User("张三", 20, "123456");

zhangsan.sayHi(); // Hi, I'm 张三

运行正常,貌似没什么问题,但其实里面还是有些坑:

父类的构造函数被执行了 2 次:继承特权成员时 1 次,继承原型时又 1 次。

父类初始化两次,这有时会导致一些问题,举个例子,父类构造函数中有个alert,那么创建子类实例时,会发现有两次弹框。

不仅如此,还导致了下面的问题。从控制台中查看子类的实例,结构如下:

 

 

可以看到子类的原型中也包含了父类的特权成员(直接创建了一个父类实例,当然会有特权成员),只不过因为解释器的属性查找机制,被子类的特权成员所覆盖,只要子类的特权成员被删除,原型中相应的成员就会暴露出来:

delete zhangsan.name;

alert(zhangsan.name); // 此时访问到的就是原型中的name

那怎么办呢?对此道爷提供了一个很实用的解决方案—— 原型式寄生组合继承。

3.3、原型式寄生组合继承

我们的目的是子类原型只继承父类的原型,而不要特权成员,原理其实很简单:创建一个临时的类,让其原型指向父类原型,然后将子类原型指向该临时类的实例即可。实现如下:

function inheritPrototype(subClass, superClass) {

function Temp() {}

Temp.prototype = superClass.prototype;

subClass.prototype = new Temp();

subClass.prototype.constructor = subClass;

}

inheritPrototype(User, Person);

因为临时类的构造函数是空实现,子类在继承原型时自然不会执行到父类的初始化操作,也不会继承到一堆乱七八糟的特权成员。

再看下 zhangsan 的结构:

 

 

此时,子类实例的原型链大致如下:

 

 

总结

修改后的代码整理如下:

// 用于子类继承父类原型的工具函数

function inheritPrototype(subClass, superClass) {

function Temp() {}

Temp.prototype = superClass.prototype;

subClass.prototype = new Temp();

subClass.prototype.constructor = subClass;

}

// 父类

function Person(name, age) {

// 特权成员(每个实例都有一份副本)

this.name = name;

this.age = age;

}

// 原型成员(所有实例共享)

Person.prototype.sayHi = function() {

alert("Hi, I'm " + this.name);

};

// 子类

function User(name, age, password) {

// 继承父类特权成员(在子类中执行父类的初始化操作)

Person.call(this, name, age);

// 添加新的特权成员

this.password = password;

}

// 继承父类原型

inheritPrototype(User, Person);

// 重写父类原型方法

User.prototype.sayHi = function() {

alert("Hi, my name is " + this.name);

};

// 扩展子类原型

User.prototype.getPassword = function() {

return this.password;

};

到此为止,我们已经比较完美地实现了类和类的继承。

但每个类、每个子类、每个子类的子类,都要这么分几步写,也是很蛋疼的。对象有对象工厂,类当然也可以搞个 类工厂(Class Factory),江湖上已有不少现成的类工厂,让我们可以从统一规范的入口来生成自定义类。

免责声明:本文来源于博客园,由网友提供或网络搜集,仅供个人研究、交流学习使用,不涉及商业盈利目的。如有版权问题,请联系本站管理员予以更改或删除。优知网会定期发布Web前端相关趋势文章,包括 Web标准 HTML5 CSS JS JQuery Ajax 网页特效 等领域,敬请关注!

0 0
原创粉丝点击
热门问题 老师的惩罚 人脸识别 我在镇武司摸鱼那些年 重生之率土为王 我在大康的咸鱼生活 盘龙之生命进化 天生仙种 凡人之先天五行 春回大明朝 姑娘不必设防,我是瞎子 浓硫酸沾到皮肤上怎么办 浓硫酸溅到皮肤上怎么办 浓硫酸滴到皮肤上怎么办 浓硫酸洒在皮肤上怎么办 浓硫酸溅到眼睛里怎么办 盐酸弄到眼睛了怎么办 稀硫酸进眼睛里怎么办 草酸弄到皮肤上怎么办 大理石被盐酸烧发白怎么办 香薰蜡烛化了怎么办 吸入了大量燃烧纸气体怎么办 狗链条上锈了怎么办 思维迟钝反应慢嘴笨怎么办 小孩思维慢反应迟钝怎么办 苹果4g网络慢怎么办 医院没有号了怎么办啊 fgo宝具动画卡顿怎么办 死刑犯在执行前死亡怎么办 汕头交警 违章扣分怎么办办理 幼儿园家长不保险应该怎么办 csgo掉白银坑了怎么办 错过教资认定现场确认怎么办 乡村建设导致民房开裂怎么办 项目部公章丢了怎么办 手机掉了没有卡怎么办 苹果系统软件删了还是出现怎么办 钉钉检测到作弊怎么办 电脑麦说话声音小怎么办 穿越火线麦克风有杂音怎么办 手闲不住就抠东西怎么办 大便堵在肛门口怎么办 在外地流量不够用怎么办 电脑键盘数字键没反应怎么办 音响音量键坏了怎么办 摩托罗拉移动电话通话没声音怎么办 摩托罗拉对讲机充电座坏了怎么办 手机导航键太灵敏怎么办 经侦大队不立案怎么办 槐茂酱菜太咸怎么办 法院执法局执法不公怎么办 搞养殖卖不出去怎么办