Prototype - 不一样的 Class

我第一次接触面向对象时,使用的是 Python,运用 def __init__(self, *args): 这种 magic method 来完成自身参数的初始化。 JS 当中也有类似的构造器 constructor 。

我先用 ES6 中新引入的 class 语法来定义一个类,因为这样容易理解。

class Student {
  constructor(name) {
    this.name = name;
  }
  
  hello() {
    return 'Hello, ' + this.name + '!';
  }
}

再定义一个类来继承

class PrimaryStudent extends Student {
  constructor(name, grade) {
    super(name); // 调用父类的构造函数
    this.grade = grade;
  }
  
  myGrade() {
    return 'I am at grade ' + this.grade;
  }
}

看似与 Python 极为相似,难怪我直接用 React 写应用的时候并没有发现不对。这种新的语法简化了 JS 原本复杂的原型 (prototype) 继承,也让直接了解新语法的我忽略了它的本质及特性。

(为了学习)回归原本的面貌

function Student(name) {
  this.name = name;
}

Student.prototype.hello = function() {
  return 'Hello, ' + this.name + '!';
}

var kitty = new Student('kitty');
var daisy = new Student('daisy');

可以观察到我们并不在 Student 中直接写 hello 函数,而是写在了它的 prototype 中。只要我们给 prototype 定义了函数,那么 Student 生成的对象就都可以调用这个函数,省去每个对象再独自生成函数的操作。

我们把 Student.prototype 想象成一根树枝,kitty 和 daisy 就是生出的绿叶了,它们共用树枝上的器官,不必自己再生出相同的器官来消耗资源。此时 kitty 和 daisy 的原型指向 Student.prototype。

Javascript规定,每一个构造函数都有一个prototype属性,指向另一个对象。这个对象的所有属性和方法,都会被构造函数的实例继承。

进一步,为了避免生成对象的时候漏写 new,可以封装一下。

function Student(props) {
    this.name = props.name || '匿名'; // 默认值为'匿名'
    this.grade = props.grade || 1; // 默认值为1
}

Student.prototype.hello = function () {
    alert('Hello, ' + this.name + '!');
};

function createStudent(props) {
    return new Student(props || {});  // 兼顾没有参数的情况
}

然后直接调用 createStudent() 就可以了。

Student.prototype 的继承应该如何写呢?

function PrimaryStudent(props) {
    // 调用Student constructor构造函数,绑定this变量
    Student.call(this, props);
    this.grade = props.grade || 1;
}

别以为这就结束了。

Student.call(this, props); 这句代码,确实和 Student 搭上边了,但此时 PrimaryStudent.prototype 是谁?

我们的目标是构造一个 PrimaryStudent.prototype ,令它指向上层的 Student.prototype。

有谁是这种关系?kitty 和 daisy 的原型好像就是。

我就不想了,前辈的方法是这样的。

function F() {
}

F.prototype = Student.prototype;

PrimaryStudent.prototype = new F();

// 把PrimaryStudent原型的构造函数修复为PrimaryStudent(因为上一句修改了constructor)
PrimaryStudent.prototype.constructor = PrimaryStudent;
  • 构造一个函数 F ,令 F.prototype = Student.prototype
  • 生成一个中间对象,那么这个对象的 prototype 就可以指向 Student.prototype

这不就是我们想要的吗?我们把该对象赋给 PrimaryStudent.prototype ,那么就可以实现 PrimaryStudent.prototype —> Student.prototype 的指向连接了。

// 继续在PrimaryStudent原型(就是new F()对象)上定义方法:
PrimaryStudent.prototype.getGrade = function () {
    return this.grade;
};

// 创建xiaoming:
var xiaoming = new PrimaryStudent({
    name: '小明',
    grade: 2
});
xiaoming.name; // '小明'
xiaoming.grade; // 2

// 验证原型:
xiaoming.__proto__ === PrimaryStudent.prototype; // true
xiaoming.__proto__.__proto__ === Student.prototype; // true

// 验证继承关系:
xiaoming instanceof PrimaryStudent; // true
xiaoming instanceof Student; // true

封装它

function extend(Child, Parent) {
  var F = function(){};
  F.prototype = Parent.prototype;
  Child.prototype = new F();
  Child.prototype.constructor = Child;
  Child.uber = Parent.prototype;
}

Child.uber = Parent.prototype; 意思是为子对象设一个uber属性,这个属性直接指向父对象的prototype属性。(uber是一个德语词,意思是"向上"、"上一层"。)这等于在子对象上打开一条通道,可以直接调用父对象的方法。这一行放在这里,只是为了实现继承的完备性,纯属备用性质。

参考资料

廖雪峰的三篇文章(个人感觉有些晦涩,最好再去看看别的资料)

推荐阮一峰的三篇文章,由简入深,非常不错。

http://www.ruanyifeng.com/blog/2010/05/object-oriented_javascript_encapsulation.html

http://www.ruanyifeng.com/blog/2010/05/object-oriented_javascript_inheritance.html

http://www.ruanyifeng.com/blog/2010/05/object-oriented_javascript_inheritance_continued.html

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,012评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,628评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,653评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,485评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,574评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,590评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,596评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,340评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,794评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,102评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,276评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,940评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,583评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,201评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,441评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,173评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,136评论 2 352

推荐阅读更多精彩内容