zoukankan      html  css  js  c++  java
  • [转]JavaScript ES6 class指南

    [转]JavaScript ES6 class指南

    前言

    EcmaScript 2015 (又称ES6)通过一些新的关键字,使类成为了JS中一个新的一等公民。但是目前为止,这些关于类的新关键字仅仅是建立在旧的原型系统上的语法糖,所以它们并没有带来任何的新特性。不过,它使代码的可读性变得更高,并且为今后版本里更多面向对象的新特性打下了基础

    这样做的原因是为了保证向后兼容性。也就是,旧代码可以在不做任何hack的情况下,与新代码同时运行。

    定义类

    让我们回想一下在ES5中定义一个类的方式。通过不是很常用的Object.defineProperty方法,我可以定义一些只读的属性。

    function Vehicle(make, year) {
      Object.defineProperty(this, 'make', {
        get: function() { return make; }
      });
    
      Object.defineProperty(this, 'year', {
        get: function() { return year; }
      });
    }
    
    Vehicle.prototype.toString = function() {
      return this.make + ' ' + this.year;
    }
    
    var vehicle = new Vehicle('Toyota Corolla', 2009);
    
    console.log(vehicle.make); // Toyota Corolla
    vehicle.make = 'Ford Mustang';
    console.log(vehicle.toString()) // Toyota Corolla 2009
    

    很简单,我们定义了一个有两个只读属性和一个自定义toString方法的Vehicle类。让我们在ES6中来做一样的事情:

    class Vehicle {
      constructor(make, year) {
        this._make = make;
        this._year = year;
      }
    
      get make() {
        return this._make;
      }
    
      get year() {
        return this._year;
      }
    
      toString() {
        return `${this.make} ${this.year}`;
      }
    }
    
    var vehicle = new Vehicle('Toyota Corolla', 2009);
    
    console.log(vehicle.make); // Toyota Corolla
    vehicle.make = 'Ford Mustang';
    console.log(vehicle.toString()) // Toyota Corolla 2009
    

    类声明

    在ES6中,有两个声明类的方式。第一种方法叫作 类声明,这也是我们在上述例子中使用的方式。

    class Vehicle {
        hello() {
            console.log('nice to meet you');
        }
    }
    

    有一个需要注意的地方是,类声明与函数声明不同,它不会被提升(hoisted)。例如,以下的代码工作正常:

    console.log(helloWorld());
    
    function helloWorld() {
      return "Hello World";
    }
    

    但是,以下代码会抛出一个异常:

    var vehicle = new Vehicle();
    
    class Vehicle {
    }
    

    类表达式

    另一个定义类的方式叫做 类表达式。它与函数表达式的运行方式完全一样。一个类表达式可以是具名的也可以是匿名的。

    var Vehicle = class {
    }
    
    var Vehicle = class VehicleClass {
      constructor() {
        // VehicleClass is only available inside the class itself
      }
    }
    
    console.log(VehicleClass); // throws an exception
    

    静态方法

    static关键字是ES6的另一个语法糖,它使静态方法声明也成为了一个一等公民。在ES5中,静态方法就像是构造函数的一个属性。

    function Vehicle() {
      // ...
    }
    
    Vehicle.compare = function(a, b) {
      // ...
    }
    

    在使用了新的static关键字后:

    class Vehicle {
      static compare(a, b) {
        // ...
      }
    }
    

    在底层,JavaScript所做的,也只是将这个方法添加为Vehicle构造函数的一个属性。值得注意的是,你也可以用同样的语法为类添加静态属性。

    类继承

    旧的原型继承有时看起来让人非常头疼。ES6中新的extends关键字解决了这个问题。在ES5,我们是这么做的:

    function Motorcycle(make, year) {
      Vehicle.apply(this, [make, year]);
    }
    
    Motorcycle.prototype = Object.create(Vehicle.prototype, {
      toString: function() {
        return 'Motorcycle ' + this.make + ' ' + this.year;
      }
    });
    
    Motorcycle.prototype.constructor = Motorcycle;
    

    使用的新的extends关键字,看上去就清晰多了:

    class Motorcycle extends Vehicle {
      constructor(make, year) {
        super(make, year);
      }
    
      toString() {
        return `Motorcycle ${this.make} ${this.year}`;
      }
    }
    

    super关键字也可以用于静态方法:

    class Vehicle {
      static compare(a, b) {
        // ...
      }
    }
    
    class Motorcycle extends Vehicle {
      static compare(a, b) {
        if (super.compare(a, b)) {
          // ...
        }
      }
    }
    

    上一个例子也展示了新的super关键字的用法。当你想要调用父类的函数时,这个关键字就显得十分好用。

    在想要调用父类的构造函数时,你可以简单地将super关键字视作一个函数使用,如super(make, year)。对于父类的其他函数,你可以将super视作一个对象,如super.toString()。例子:kk:super指向原型对象

    class Motorcycle extends Vehicle {
      toString() {
        return 'Motorcycle ' + super.toString();
      }
    }
    

    可被计算的方法名

    当在class中声明属性时,定义属性名时,你可以使用表达式。这个语法特性在一些ORM类库中将会非常流行。例子:

    function createInterface(name) {
      return class {
        ['findBy' + name]() {
          return 'Found by ' + name;
        }
      }
    }
    
    const Interface = createInterface('Email');
    const instance = new Interface();
    
    console.log(instance.findByEmail());
    

    最后

    在当前,使用class关键字来声明类,而不使用原型,获得的仅仅是语法上的优势。但是,这个是一个适应新语法和新实践的好开始。JavaScript每天都在变得更好,并且通过class关键字,可以使各种工具更好得帮助你。

  • 相关阅读:
    Jmeter-日期格式转换为时间戳
    Java中异常的处理
    Java中接口的新特性,为接口添加静态方法和默认方法
    Java接口的应用之代理模式
    Java之接口(interface)的理解
    JAVA中使用super调用属性、方法、构造器
    JAVA方法的重写
    Java之方法
    Java之匿名对象
    关于多态性的使用
  • 原文地址:https://www.cnblogs.com/stephenykk/p/6247468.html
Copyright © 2011-2022 走看看