JavaScript发展到今天,和其他语言不一样的一个特点是,有各种各样的"继承方式",或者稍微准确一点的说法,叫做有各种各样的基于prototype的模拟类继承实现方式。
在ES6之前,JavaScript没有类继承的概念,因此使用者为了代码复用的目的,只能参考其他语言的"继承",然后用prototype来模拟出对应的实现,于是有了各种继承方式,比如《JavaScript高级程序设计》上说的 原型链,借用构造函数,组合继承,原型式继承,寄生式继承,寄生组合式继承 等等
那么多继承方式,让第一次接触这一块的小伙伴们内心有点崩溃。然而,之所以有那么多继承方式,其实还是因为"模拟"二字,因为我们在说继承的时候不是在研究prototype本身,而是在用prototype和JS特性来模拟别的语言的类继承。
我们现在抛开这些种类繁多的继承方式,来看一下prototype的本质和我们为什么要模拟类继承。
"原型" 这个词本身源自心理学,指神话、宗教、梦境、幻想、文学中不断重复出现的意象,它源自民族记忆和原始经验的集体潜意识。
所以,原型是一种抽象,代表事物表象之下的联系,用简单的话来说,就是原型描述事物与事物之间的相似性.
想象一个小孩子如何认知这个世界:
当小孩子没见过老虎的时候,大人可能会教他,老虎啊,就像是一只大猫。如果这个孩子碰巧常常和邻居家的猫咪玩耍,那么她不用去动物园见到真实的老虎,就能想象出老虎大概是长什么样子。
这个故事有个更简单的表达,叫做"照猫画虎"。如果我们用JavaScript的原型来描述它,就是:
function Tiger(){
//...
}
Tiger.prototype = new Cat(); //老虎的原型是一只猫
很显然,"照猫画虎"(或者反过来"照虎画猫",也可以,取决孩子于先认识老虎还是先认识猫)是一种认知模式,它让人类儿童不需要在脑海里重新完全构建一只老虎的全部信息,而可以通过她熟悉的猫咪的"复用"得到老虎的大部分信息,接下来她只需要去到动物园,去观察老虎和猫咪的不同部分,就可以正确认知什么是老虎了。这段话用JavaScript可以描述如下:
function Cat(){
}
//小猫喵喵叫
Cat.prototype.say = function(){
return "喵";
}
//小猫会爬树
Cat.prototype.climb = function(){
return "我会爬树";
}
function Tiger(){
}
Tiger.prototype = new Cat();
//老虎的叫声和小猫不同,但老虎也会爬树
Tiger.prototype.say = function(){
return "嗷";
}
所以,原型可以通过描述两个事物之间的相似关系来复用代码,我们可以把这种复用代码的模式称为原型继承。
几年之后,当时的小孩子长大了,随着她的知识结构不断丰富,她认识世界的方式也发生了一些变化,她学会了太多的动物,有喵喵叫的猫,百兽之王狮子,优雅的丛林之王老虎,还有豺狼、大象等等。
这时候,单纯的相似性的认知方式已经很少被使用在如此丰富的知识内容里,更加严谨的认知方式----分类,开始被更频繁使用。
这时候当年的小孩会说,猫和狗都是动物,如果她碰巧学习的是专业的生物学,她可能还会说猫和狗都是脊索门哺乳纲,于是,相似性被"类"这一种更高程度的抽象表达取代,我们用JavaScript来描述:
class Animal{
eat(){}
say(){}
climb(){}
...
}
class Cat extends Animal{
say(){return "喵"}
}
class Dog extends Animal{
say(){return "汪"}
}
所以,原型继承和类继承是两种认知模式,本质上都是为了抽象(复用代码)。相对于类,原型更初级且更灵活。因此当一个系统内没有太多关联的事物的时候,用原型明显比用类更灵活便捷。
原型继承的便捷性表现在系统中对象较少的时候,原型继承不需要构造额外的抽象类和接口就可以实现复用。(如系统里只有猫和狗两种动物的话,没必要再为它们构造一个抽象的"动物类")
原型继承的灵活性还表现在复用模式更加灵活。由于原型和类的模式不一样,所以对复用的判断标准也就不一样,例如把一个红色皮球当做一个太阳的原型,当然是可以的(反过来也行),但显然不能将"恒星类"当做太阳和红球的公共父类(倒是可以用"球体"这个类作为它们的公共父类)。
既然原型本质上是一种认知模式可以用来复用代码,那我们为什么还要模拟"类继承"呢?在这里面我们就得看看原型继承有什么问题----
由于我们刚才前面举例的猫和老虎的构造器没有参数,因此大家很可能没发现问题,现在我们试验一个有参数构造器的原型继承:
function Vector2D(x, y){
this.x = x;
this.y = y;
}
Vector2D.prototype.length = function(){
return Math.sqrt(this.x * this.x + this.y * this.y);
}
function Vector3D(x, y, z){
Vector2D.call(this, x, y);
this.z = z;
}
Vector3D.prototype = new Vector2D();
Vector3D.prototype.length = function(){
return Math.sqrt(this.x * this.x + this.y * this.y + this.z * this.z);
}
var p = new Vector3D(1, 2, 3);
console.log(p.x, p.y, p.z, p.length(), p instanceof Vector2D);
上面这段代码里面我们看到我们用 Vector2D 的实例作为 Vector3D 的原型,在 Vector3D 的构造器里面我们还可以调用 Vector2D 的构造器来初始化 x、y。
但是,如果认真研究上面的代码,会发现一个小问题,在中间描述原型继承的时候:
Vector3D.prototype = new Vector2D();
我们其实无参数地调用了一次 Vector2D 的构造器!
这一次调用是不必要的,而且,因为我们的 Vector2D 的构造器足够简单并且没有副作用,所以我们这次无谓的调用除了稍稍消耗了性能之外,并不会带来太严重的问题。
但在实际项目中,我们有些组件的构造器比较复杂,或者操作DOM,那么这种情况下无谓多调用一次构造器,显然是有可能造成严重问题的。
于是,我们得想办法克服这一次多余的构造器调用,而显然,我们发现我们可以不必要这一次多余的调用:
function createObjWithoutConstructor(Class){
function T(){};
T.prototype = Class.prototype;
return new T();
}
上面的代码中,我们通过创建一个空的构造器T,引用父类Class的prototype,然后返回new T( ),来巧妙地避开Class构造器的执行。这样,我们确实可以绕开父类构造器的调用,并将它的调用时机延迟到子类实例化的时候(本来也应该这样才合理)。
function Vector2D(x, y){
this.x = x;
this.y = y;
}
Vector2D.prototype.length = function(){
return Math.sqrt(this.x * this.x + this.y * this.y);
}
function Vector3D(x, y, z){
Vector2D.call(this, x, y);
this.z = z;
}
Vector3D.prototype = createObjWithoutConstructor(Vector2D);
Vector3D.prototype.length = function(){
return Math.sqrt(this.x * this.x + this.y * this.y + this.z * this.z);
}
var p = new Vector3D(1, 2, 3);
console.log(p.x, p.y, p.z, p.length(), p instanceof Vector2D);
这样,我们解决了父类构造器延迟构造的问题之后,原型继承就比较适用了,并且这样简单处理之后,使用起来还不会影响 instanceof 返回值的正确性,这是与其他模拟方式相比最大的好处。
最后,我们利用这个原理还可以实现比较完美的类继承:
(function(global){"use strict"
Function.prototype.extend = function(props){
var Super = this; //父类构造函数
//父类原型
var TmpCls = function(){
}
TmpCls.prototype = Super.prototype;
var superProto = new TmpCls();
//父类构造器wrapper
var _super = function(){
return Super.apply(this, arguments);
}
var Cls = function(){
if(props.constructor){
//执行构造函数
props.constructor.apply(this, arguments);
}
//绑定 this._super 的方法
for(var i in Super.prototype){
_super[i] = Super.prototype[i].bind(this);
}
}
Cls.prototype = superProto;
Cls.prototype._super = _super;
//复制属性
for(var i in props){
if(i !== "constructor"){
Cls.prototype[i] = props[i];
}
}
return Cls;
}
function Animal(name){
this.name = name;
}
Animal.prototype.sayName = function(){
console.log("My name is "+this.name);
}
var Programmer = Animal.extend({
constructor: function(name){
this._super(name);
},
sayName: function(){
this._super.sayName(name);
},
program: function(){
console.log("I\"m coding...");
}
});
//测试我们的类
var animal = new Animal("dummy"),
akira = new Programmer("akira");
animal.sayName();//输出 'My name is dummy'
akira.sayName();//输出 'My name is akira'
akira.program();//输出 'I"m coding...'
})(this);
可以比较一下ES6的类继承:
(function(global){"use strict"
//类的定义
class Animal {
//ES6中新型构造器
constructor(name) {
this.name = name;
}
//实例方法
sayName() {
console.log("My name is "+this.name);
}
}
//类的继承
class Programmer extends Animal {
constructor(name) {
//直接调用父类构造器进行初始化
super(name);
}
sayName(){
super.sayName();
}
program() {
console.log("I\"m coding...");
}
}
//测试我们的类
var animal = new Animal("dummy"),
akira = new Programmer("akira");
animal.sayName();//输出 'My name is dummy'
akira.sayName();//输出 'My name is akira'
akira.program();//输出 'I"m coding...'
})(this);
原型继承和类继承是两种不同的认知模式,原型继承在对象不是很多的简单应用模型里比类继承更加灵活方便。然而JavaScript的原型继承在语法上有一个构造器额外调用的问题,我们只要通过 createObjWithoutConstructor 来延迟构造器的调用,就能解决这个问题。
京东创始人刘强东和其妻子章泽天最近成为了互联网舆论关注的焦点。有关他们“移民美国”和在美国购买豪宅的传言在互联网上广泛传播。然而,京东官方通过微博发言人发布的消息澄清了这些传言,称这些言论纯属虚假信息和蓄意捏造。
日前,据博主“@超能数码君老周”爆料,国内三大运营商中国移动、中国电信和中国联通预计将集体采购百万台规模的华为Mate60系列手机。
据报道,荷兰半导体设备公司ASML正看到美国对华遏制政策的负面影响。阿斯麦(ASML)CEO彼得·温宁克在一档电视节目中分享了他对中国大陆问题以及该公司面临的出口管制和保护主义的看法。彼得曾在多个场合表达了他对出口管制以及中荷经济关系的担忧。
今年早些时候,抖音悄然上线了一款名为“青桃”的 App,Slogan 为“看见你的热爱”,根据应用介绍可知,“青桃”是一个属于年轻人的兴趣知识视频平台,由抖音官方出品的中长视频关联版本,整体风格有些类似B站。
日前,威马汽车首席数据官梅松林转发了一份“世界各国地区拥车率排行榜”,同时,他发文表示:中国汽车普及率低于非洲国家尼日利亚,每百户家庭仅17户有车。意大利世界排名第一,每十户中九户有车。
近日,一项新的研究发现,维生素 C 和 E 等抗氧化剂会激活一种机制,刺激癌症肿瘤中新血管的生长,帮助它们生长和扩散。
据媒体援引消息人士报道,苹果公司正在测试使用3D打印技术来生产其智能手表的钢质底盘。消息传出后,3D系统一度大涨超10%,不过截至周三收盘,该股涨幅回落至2%以内。
9月2日,坐拥千万粉丝的网红主播“秀才”账号被封禁,在社交媒体平台上引发热议。平台相关负责人表示,“秀才”账号违反平台相关规定,已封禁。据知情人士透露,秀才近期被举报存在违法行为,这可能是他被封禁的部分原因。据悉,“秀才”年龄39岁,是安徽省亳州市蒙城县人,抖音网红,粉丝数量超1200万。他曾被称为“中老年...
9月3日消息,亚马逊的一些股东,包括持有该公司股票的一家养老基金,日前对亚马逊、其创始人贝索斯和其董事会提起诉讼,指控他们在为 Project Kuiper 卫星星座项目购买发射服务时“违反了信义义务”。
据消息,为推广自家应用,苹果现推出了一个名为“Apps by Apple”的网站,展示了苹果为旗下产品(如 iPhone、iPad、Apple Watch、Mac 和 Apple TV)开发的各种应用程序。
特斯拉本周在美国大幅下调Model S和X售价,引发了该公司一些最坚定支持者的不满。知名特斯拉多头、未来基金(Future Fund)管理合伙人加里·布莱克发帖称,降价是一种“短期麻醉剂”,会让潜在客户等待进一步降价。
据外媒9月2日报道,荷兰半导体设备制造商阿斯麦称,尽管荷兰政府颁布的半导体设备出口管制新规9月正式生效,但该公司已获得在2023年底以前向中国运送受限制芯片制造机器的许可。
近日,根据美国证券交易委员会的文件显示,苹果卫星服务提供商 Globalstar 近期向马斯克旗下的 SpaceX 支付 6400 万美元(约 4.65 亿元人民币)。用于在 2023-2025 年期间,发射卫星,进一步扩展苹果 iPhone 系列的 SOS 卫星服务。
据报道,马斯克旗下社交平台𝕏(推特)日前调整了隐私政策,允许 𝕏 使用用户发布的信息来训练其人工智能(AI)模型。新的隐私政策将于 9 月 29 日生效。新政策规定,𝕏可能会使用所收集到的平台信息和公开可用的信息,来帮助训练 𝕏 的机器学习或人工智能模型。
9月2日,荣耀CEO赵明在采访中谈及华为手机回归时表示,替老同事们高兴,觉得手机行业,由于华为的回归,让竞争充满了更多的可能性和更多的魅力,对行业来说也是件好事。
《自然》30日发表的一篇论文报道了一个名为Swift的人工智能(AI)系统,该系统驾驶无人机的能力可在真实世界中一对一冠军赛里战胜人类对手。
近日,非营利组织纽约真菌学会(NYMS)发出警告,表示亚马逊为代表的电商平台上,充斥着各种AI生成的蘑菇觅食科普书籍,其中存在诸多错误。
社交媒体平台𝕏(原推特)新隐私政策提到:“在您同意的情况下,我们可能出于安全、安保和身份识别目的收集和使用您的生物识别信息。”
2023年德国柏林消费电子展上,各大企业都带来了最新的理念和产品,而高端化、本土化的中国产品正在不断吸引欧洲等国际市场的目光。
罗永浩日前在直播中吐槽苹果即将推出的 iPhone 新品,具体内容为:“以我对我‘子公司’的了解,我认为 iPhone 15 跟 iPhone 14 不会有什么区别的,除了序(列)号变了,这个‘不要脸’的东西,这个‘臭厨子’。