es6语言特性的总结(3)

es6语言特性的总结(1)在这里
es6语言特性的总结(2)在这里

在ES5中,由于没有类的概念,所以如果要使用面向对象编程的方式,就需要利用原型继承的方式。通常是创建一个构造器,然后将方法指派到该构造器的原型上。
就像这样:

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

Cat.prototype.speak = function() {
  console.log('Mew!');
}

ES6引入了class关键字后就不再需要这样做了。不过需要明白的是ES6中的类仅仅是以上面这种方式作为基础的一个语法糖而已。
ES6中类声明已class关键字开始,其后是类的名称;剩余部分的语法部分看起来就像对象字面量中的方法简写,并且在方法之间不需要使用逗号。同时允许你在其中使用特殊的 constructor 方法名称直接定义一个构造器,而不需要先定义一个函数再把它当作构造器使用。

class Cat {
  constructor(name) {
    this.name = name;
  }
  
  speak() {
    console.log('Mew!');
  }
}

类声明与ES5仿类的区别

虽然ES6的类声明是ES5方式的一个语法糖,但是与之相比,还是存在一些区别的。

  1. 类声明不会被提升,这与函数定义不同。类声明的行为与 let 相似,因此在程序的执行到达声明处之前,类会存在于暂时性死区内。
  2. 类声明中的所有代码会自动运行在严格模式下,并且也无法退出严格模式。
  3. 类的所有方法都是不可枚举的,这是对于自定义类型的显著变化,后者必须用 Object.defineProperty() 才能将方法改变为不可枚举。
  4. 类的所有方法内部都没有 [[Construct]] ,因此使用 new 来调用它们会抛出错误。
  5. 调用类构造器时不使用 new ,会抛出错误。
  6. 试图在类的方法内部重写类名,会抛出错误。

访问器属性

自有属性需要在类构造器中创建,而类还允许你在原型上定义访问器属性。

class Person {
  constructor(name, age) {
    this.age = age;
    this.name = name;
  }
  
  get firstName() {
    return this.name.split(' ')[0];
  }
  
  set firstName(value) {
    let lastName = this.name.split(' ')[1];
    this.name = value + ' ' + lastName;
  }
}

let person = new Person('Michael Jackson', 35);
console.log(person.firstName); //'Michael'
person.firstName = 'Marry';
console.log(person.name); // 'Marry Jackson'

在读取访问器属性的时候,会调用getter方法,而写入值的时候,会调用setter方法。这类似于ES5中使用Object.definePropery的方法。

静态成员

静态成员在ES5中一般是直接定义在构造器上的,如:

function Person(name, age) {
  this.name = name;
  this.age = age;
}

Person.createAdult = function(name) {
  return new Person(name, 18);
};

而在ES6中提供了static关键字简化了声明静态成员的方式:

class Person {
  constructor(name, age) {
    this.age = age;
    this.name = name;
  }
  
  static createAdult(name) {
    return new Person(name, 18);
  }
   
}

继承

ES5中实现继承的方式有很多种,但是如果要实现严格的继承,步骤较为繁琐。为了简化继承的关系,ES6中使用类让这项工作变得更简单。如果你熟悉面向对象语言,如java等,那么extends这个关键你一定不会陌生。同样的,在ES6中使用extends 关键字来指定当前类所需要继承的函数即可。生成的类的原型会被自动调整,而你还能调用 super() 方法来访问基类的构造器。

class Person {
    constructor(country) {
      this.country = country;
    }
}

class Chinese extends Person{
    constructor() {
      super('China');
    }
    
    speak() {
      console.log('I come from ' + this.country);
    }
}

派生类中的方法总是会屏蔽基类中的同名方法,因此,如果你需要使用父类中定义的方法的话,可以使用super关键字来进行访问。如:

class Person {
    constructor(country) {
      this.country = country;
    }
    
    speak() {
      console.log('I come from ' + this.country);
    }
}

class Chinese extends Person{
    constructor() {
      super('China');
    }
    
    speak() {
        super.speak();
        console.log('I am a Chinese');
    }
}

const chinese = new Chinese();
chinese.speak();
//I come from China.
//I am a Chinese.

从表达式中派生类

另一个在ES6中比较高级的地方是,可以从表达式中派生出类来:

let SerializableMixin = {
    serialize() {
        return JSON.stringify(this);
    }
};

let AreaMixin = {
    getArea() {
        return this.length * this.width;
    }
};

//混入
function mixin(...mixins) {
    var base = function() {};
    Object.assign(base.prototype, ...mixins);
    return base;
}

class Square extends mixin(AreaMixin, SerializableMixin) {
    constructor(length) {
        super();
        this.length = length;
        this.width = length;
    }
}

var x = new Square(3);
console.log(x.getArea());               // 9
console.log(x.serialize());             // "{"length":3,"width":3}"

继承内置对象

利用extends继承内置对象的时候,容易出现的一个问题是会返回内置对象实例的方式,在继承后会返回子类的实例。如:

class SubArray extends Array {
  
}

const subArr = new SubArray(1,2,3);
const filteredArr = subArr.filter(value => value > 1); 
console.assert(filteredArr instanceof SubArray);  //true

如果需要想让其返回实例类型是Array可以利用Symbol.species这个符号来处理:

class SubArray extends Array {
//这里使用static,表明是静态访问器属性
  static get [Symbol.species]() {
    return Array;
  }
}

定义抽象类

利用之前介绍的new.target可以实现一个抽象类,原理就是当用户调用new直接创建实例的时候,抛出错误。:

class BaseClass {
  constructor() {
    if(new.target === BaseClass) {
      throw new Error('该类不能直接实例化')
    }
  }
}

模块

随着项目的规模越来越大,现在模块化已经成为开发过程中必备的流程。之前,我们可能借助RequireJS等工具进行模块化管理,而现在ES6已经提供了模块系统。

先来了解一下基本语法:

基本的导出导入

模块( Modules )本质上就是 包含JS 代码的文件。在一个js文件中,你可以使用export关键字,将代码公开给其他模块。

// sayHello.js
export function sayHello() {
  console.log('hello');
}

// funcs.js
export function fun1() { .... }
export function func2() { .... }
export const value1 = 'value1';

如上面的例子中所示,你可以在文件中导出所有的最外层函数以及varletconst声明的变量。而这些导出的变量或公开部分则可以被其他文件利用import语法进行导入后引用。

//单个导入
import {sayHello} from './sayHello.js';
//多个导入
import {func1, func2} from './funs.js';
sayHello(); // hello

为了确保浏览器与Node.js之间保持良好的兼容性,建议使用相对路径的写法。

如果需要将整个模块当做单一的对象进行导入,可以使用*通配符:

//使用as关键字为导出对象设置别名,模块中所有导出都将作为属性存在
import * as funcs from './funcs.js';

funcs.func1();
funsc.func2();

重命名导出与导入

如果不想用原来模块中的命名,可以通过as关键字来指定别名。

//as前面为模块原先的名称,后面是别名,使用别名后sayHello为undefined
import { sayHello as say } from './sayHello.js';

say();

默认值

你可以使用export关键字来导出默认模块:

// sayHello.js
export default function() {
  console.log('hello');
}

// main.js
import sayHello from './sayHello.js';
sayHello();

可以注意到,这里默认导出的时候,不需要使用花括号,而直接为其命名即可。这种写法也较为简洁。当一个文件中,同时存在默认导出模块和非默认导出模块的时候,导出的时候,默认导出模块需要写在前面,例如:

import sayHello,{ func1 } from './sayHello.js'; //此处略去导出过程
//或者使用如下方式
import {default as sayHello, func1} from './sayHello.js';

无绑定导出

当一个文件中没有使用export语句进行导出的时候,其实我们还是可以import进行导入的。通常是被用于创建polyfill与shim的时候。

//sayHello.js
const name = 'scq000';
function sayHello() {
    console.log('hello');
}

// main.js
import './sayHello.js';

sayHello();
console.log(name);

加载模块

虽然说现在在项目中通常都使用webpack来处理模块代码,但也需要知道其他加载模块的方式。

你可以使用

你可能感兴趣的:(es6语言特性的总结(3))