第二十章 Class的基本语法

  1. 简介

JS中,生成实例对象的传统方法是通过构造函数

function Point(x, y) {
this.x = x;
this.y = y;
}

Point.prototype.toString = function () {
return '(' + this.x + ', ' + this.y + ')';
};

var p = new Point(1, 2);

上面这种写法跟传统的面向对象语言(比如 C++ 和 Java)差异很大,很容易让新学习这门语言的程序员感到困惑。

ES6 提供了更接近传统语言的写法,引入了 Class(类)这个概念,作为对象的模板。通过class关键字,可以定义类。

基本上,ES6 的class可以看作只是一个语法糖,它的绝大部分功能,ES5 都可以做到,新的class写法只是让对象原型的写法更加清晰、更像面向对象编程的语法而已。
上面的代码用 ES6 的class改写,就是下面这样。

// 定义类
class Point{
constructor(x, y){
this.x = x; // this 代表实例对象
this.y = y;
}

toString() {
    return '(' + this.x + ', ' + this.y + ')';
}

}

上面代码定义了一个“类”,可以看到里面有一个 constructor 方法,这就是构造方法,而this关键字则代表实例对象。也就是说,ES5 的构造函数Point,对应 ES6 的Point类的构造方法。

Point类除了构造方法,还定义了一个toString方法。
注意,
(1) 定义“类”的方法的时候,前面不需要加上function这个关键字,直接把函数定义放进去了就可以了。
(2) 另外,方法之间不需要逗号分隔,加了会报错。

ES6 的类,完全可以看做构造函数的另一种写法:

class Point{
constructor(x, y){
this.x = x; // this 代表实例对象
this.y = y;
}

toString() {
    return '(' + this.x + ', ' + this.y + ')';
}

}

typeof Point // 'function'
Point === Point.prototype.constructor // true
Point.prototype.toString

使用的时候,也是直接对类使用 new 命令,跟构造函数的用法完全一致。
class Bar {
doStuff() {
console.log('stuff');
}
}

var b = new Bar();
b.doStuff(); // 'stuff'

构造函数的 prototype 属性,在 ES6 的“类”上面继续存在。事实上,类的所有方法都定义在类的 prototype 属性上面

class Point{
constructor() {
// ...
}

toString() {
    // ...
}

toValue() {
    // ...
}

}

// 等同于
Point.prototype = {
constructor() {},
toString() {},
toValue() {}
}

class 是个语法糖的本质显露无疑,就是使用 constructor 指向构造函数Point本身来实现内部属性的处理,然后其他的toString都直接绑定在原型prototype上,和ES5原理一样。

在类的实例上面调用方法,其实就是调用原型上的方法。

class B {}

let b = new B();

b.constructor === B.prototype.constructor // true

Object.assign() 方法可以很方便地一次向类添加多个方法。

class Point {
constructor() {
// ...
}
}

Object.assign(Point.prototype, {
toString() {},
toValue() {}
})

另外,类的内部所有定义的方法,都是不可枚举的

class Point {
constructor(x, y) {
// ...
}

toString() {
// ...
}
}

Object.keys(Point.prototype)
// []
Object.getOwnPropertyNames(Point.prototype)
// ["constructor","toString"]

上面代码中,toString方法是Point类内部定义的方法,它是不可枚举的。这一点与 ES5 的行为不一致。

var Point = function (x, y) {
// ...
};

Point.prototype.toString = function() {
// ...
};

Object.keys(Point.prototype)
// ["toString"]
Object.getOwnPropertyNames(Point.prototype)
// ["constructor","toString"]

类的属性名,可以采用表达式:

let methodName = 'getArea';

class Square {
constructor(length){
// ...
}

[methodName]() {
    // ...
}

}

  1. 严格模式
    类和模块的内部,默认就是严格模式。

  2. constructor 方法
    和ES5的基本一样

constructor方法默认返回实例对象(即this),完全可以指定返回另外一个对象

class Foo {
constructor() {
return Object.create(null);
}
}

new Foo() instanceof Foo // false

上面代码中,constructor函数返回一个全新的对象,结果导致实例对象不是Foo类的实例。

ES5:

function Foo(){

}

Foo.prototype.constructor = function(){
return Object.create(null);
}

new Foo() instanceof Foo // true

  1. 类的实例对象
    ES5中,如果调用构造函数时候,没有使用 new ,则就会当做普通函数来调用,这一点来说,JS设计的不严谨,必须得手动去写一个更加安全的构造函数

function Person(name, age){
if ( this instanceof Person ){
this.name = name;
this.age = age;
} else {
return new Person(name, age)
}
}

ES6中,避免了这一个问题,使用 class 的构造函数,没有使用 new 调用,就会报错

class Point {
// ...
}

// 报错
var point = Point(2, 3);

// 正确
var point = new Point(2, 3);

与 ES5 一样,实例的属性除非显式的定义在其本身(即定义在this对象上),否则都是定义在原型上(即定义在class上)
Javascript中Object对象原型上的hasOwnProperty()用来判断一个属性是定义在对象本身而不是继承自原型链

上面代码中,x和y都是实例对象point自身的属性(因为定义在this变量上),所以hasOwnProperty方法返回true,
而toString是原型对象的属性(因为定义在Point类上),所以hasOwnProperty方法返回false。这些都与 ES5 的行为保持一致。

class Point {
constructor(x, y){
this.x = x;
this.y = y;
}

toString() {
    return '(' + this.x + ', ' + this.y + ')';
}

}

var point = new Point(2,3);

point.toString(); // (2, 3)

point.hasOwnProperty('x'); // true
point.hasOwnProperty('y'); // true
point.hasOwnProperty('toString'); // false
point.proto.hasOwnProperty('toString'); // true

与 ES5 一样,类的所有实例共享一个原型对象

var p1 = new Point(2,3);
var p2 = new Point(3,2);

p1.proto === p2.proto
//true

  1. Class 表达式

与函数一样,类也可以使用表达式的形式定义:

const MyClass = class Me {
getClassName() {
return Me.name;
}
}

上面代码使用表达式定义了一个类。需要注意的是,这个类的名字是MyClass而不是Me,Me只在 Class 的内部代码可用,指代当前类

let inst = new MyClass();
inst.getClassName() // Me
Me.name // ReferenceError: Me is not defined

如果类的内部没有用到 Me, 则可以省略 Me

const MyClass = class { /* ... */ };

采用 Class 表达式,可以写出立即执行的 Class

let person = new class {
constructor(name) {
this.name = name;
}

sayName() {
    console.log(this.name);
}

}('张三');

person.sayName(); // "张三"

  1. 不存在变量提升
    类不存在变量提升,这一点和ES5不一样

new Foo(); // ReferenceError
class Foo {}

这种规定的原因和 class 继承有关,必须保证子类在父类之后定义

{
let Foo = class {};
class Bar extends Foo {
}
}

上面的代码不会报错,因为Bar继承Foo的时候,Foo已经有定义了。但是,如果存在class的提升,上面代码就会报错,因为class会被提升到代码头部,而let命令是不提升的,所以导致Bar继承Foo的时候,Foo还没有定义。

  1. 私有方法和私有属性

私有方法是常见需求,但 ES6 不提供,只能通过变通方法模拟实现。

一种做法是在命名上加以区别:

class Widget {

// 公有方法
foo (baz) {
this._bar(baz);
}

// 私有方法
_bar(baz) {
return this.snaf = baz;
}

// ...
}

上面代码中,_bar方法前面的下划线,表示这是一个只限于内部使用的私有方法。但是,这种命名是不保险的,在类的外部,还是可以调用到这个方法

另一种方法就是索性将私有方法移出模块,因为模块内部的所有方法都是对外可见的

class Widget {
foo (baz){
bar.call(this, baz);
}
// ...
}

function bar(baz){
return this.sanf = baz;
}

上面代码中,foo是公有方法,内部调用了bar.call(this, baz)。这使得bar实际上成为了当前模块的私有方法。

还有一种方法是利用 Symbol 值的唯一性,将私有方法的名字命名为一个 Symbol 值

const bar = Symbol('bar');
const snaf = Symbol('snaf');

export default class myClass {

// 公有方法
foo(baz) {
   this[bar](baz);
}

// 私有方法
[bar](baz){
    return this[snaf] = baz;
}

// ...

}

上面代码中,bar和snaf都是Symbol值,导致第三方无法获取到它们,因此达到了私有方法和私有属性的效果。

私有属性的提案

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

推荐阅读更多精彩内容