基本的类声明

类声明以class关键字开始,其后是类的名称;剩余部分的语法看起来像对象字面量中的方法简写,并且在方法之间不需要使用逗号。

class Person {
//等价于prototype的构造器
constructor(name) {
this.name = name;
} SayName() {
console.log(this.name);
}
} let per = new Person("cf");
per.SayName();//cf console.log(typeof Person);//function

构造器最大的用处就是在创建对象时执行初始化,当创建一个对象时,系统会为这个对象的实例进行默认的初始化。如果想改变这种默认的初始化,就可以通过自定义构造器来实现。

类与自定义类型的区别

  1. 类声明不会被提升。类声明的行为类似let,因此在程序的执行到达声明处之前,类会存在于暂时性死区内。
  2. 类声明中的所有代码会自动运行在严格模式下,并且也无法退出严格模式
  3. 类的所有方法都是不可枚举的
  4. 类的所有方法内部都没有[[Construct]],因此使用new来调用他们会抛出错误
  5. 调用类构造器时不使用new会抛出错误。
  6. 试图在类的内部修改重写类名,会抛出错误。

类与函数有相似之处,即它们都有两种形式:声明与表达式

基本的类表达式

let PersonClass = class {
constructor(name) {
this.name = name;
} SayName() {
console.log(this.name);
}
} let per = new PersonClass("cc");
per.SayName();

类表达式和类声明都不会被提升

具名类表达式

let PersonClass = class PersonClass2 {
constructor(name) {
this.name = name;
} SayName() {
console.log(this.name);
console.log(PersonClass2);
}
} let per = new PersonClass("cc");
per.SayName();

此例中的类表达式被命名为PersonClass2.PersonClass2只在类定义内部存在,因此只能用在类方法内部。

作为以及公民的类

在编程中,能被当做值来使用的就成为一级公民,意味着他能作为参数,能作为函数返回值,能用来给变量赋值。js的函数就是一级公民。

let createObject = function (classDef) {
return new classDef();
} let obj = new createObject(class {
SayHi() {
console.log("hi");
}
}) obj.SayHi();

自有属性需要在类构造器中创建,而类还允许你在原型上定义访问器属性。为了创建一个getter ,要使用 get 关键字,并要与后方标识符之间留出空格;创建 setter 用相同方式,只是要换用 set 关键字

class CustomHtmlElement {
constructor(element) {
this.element = element;
}
get html() {
return this.element.innerHTML;
} set html(value) {
this.element.innerHTML = value;
}
}

需计算的成员名

类方法与类访问器属性也能使用需计算的名称。语法相同于对象字面量中的需计算名称:无须使用标识符,而是用方括号来包裹一个表达式

let methodName = "SayName";
class PersonClass {
constructor(name) {
this.name = name;
} [methodName]() {
console.log(this.name);
}
} let per = new PersonClass("cc");
per.SayName();

生成器方法

可以使用Symbol.iterator来定义生成器方法,从而定义出类的默认迭代器。

class Collection {
constructor() {
this.items = [];
} *[Symbol.iterator]() {
yield this.items;
}
} let collection = new Collection(); collection.items.push(1);
collection.items.push(2);
collection.items.push(3); for (let num of collection.items) {
console.log(num);
}

静态成员

静态成员不能通过实例来访问,你始终需要直接用类自身来访问

class PersonClass {
constructor(name) {
this.name = name;
} SayName() {
console.log(this.name);
} static create() {
return new PersonClass(name);
}
} let per = new PersonClass.create("cc");

使用派生类进行继承

class Rectangle {
constructor(length, width) {
this.length = length;
this.width = width;
} getArea() {
return this.length * this.width;
}
} class Square extends Rectangle {
constructor(length) {
super(length, length);
}
} let sq = new Square(10);
console.log(sq.getArea());//100

继承了其他类的类被称为派生类。如果派生类指定了构造器,就需要使用super(),否则就会出错。如果不定义构造器,super()方法会被自动调用,并会使用创建新实例时提供的所有参数。例如:


class Square extends Rectangle { } let sq = new Square(10, 10);
console.log(sq.getArea());//100

使用super需要牢记以下几点

  1. 你只能在派生类中使用super。
  2. 在构造器中,你必须在访问this之前调用super()。由于super()负责初始化this,因此试图先访问this自然后报错。
  3. 唯一能避免调用super()的办法,是从类构造器中返回一个对象。

屏蔽类方法

派生类中的方法总是会屏蔽基类的同名方法。例如:你可以将getArea()方法添加到Square类,以便重新定义它的功能。

class Rectangle {
constructor(length, width) {
this.length = length;
this.width = width;
} getArea() {
return this.length * this.width;
}
} class Square extends Rectangle {
constructor(length, name) {
super(length, length);
this.name = name;
} getArea() {
return `this is ${this.name} input ${this.length}`
}
}

你也可以使用super.getArea()方法来调用基类中的同名方法

class Square extends Rectangle {
constructor(length) {
super(length, length);
} getArea() {
return super.getArea();
}
}

继承静态成员

如果基类包含静态成员,那么这些静态成员在派生类中也是可用的。

class Rectangle {
constructor(length, width) {
this.length = length;
this.width = width;
} getArea() {
return this.length * this.width;
} static create(length, width) {
return new Rectangle(length, width);
}
} class Square extends Rectangle {
constructor(length) {
super(length, length);
}
}
let sqr = Square.create(10, 10); console.log(sqr.getArea());//100

从表达式中派生类

在ES6中派生类的最强大能力,或许就是能够从表达式中派生类。只要一个表达式能够返回一个具有[[Constructor]]属性以及原型的函数,你就可以对其使用extends。

function Rectangle(length,width){
this.length = length;
this.width = width;
} Rectangle.prototype.getArea = function(){
return this.length * this.width;
} class Square extends Rectangle{
constructor(length){
super(length,length);
}
}
let x = new Square(10); console.log(x.getArea());//100

extends后面能接受任意类型的表达式,这带来了巨大的可能性,例如动态的决定要继承的类。

function Rectangle(length, width) {
this.length = length;
this.width = width;
} Rectangle.prototype.getArea = function () {
return this.length * this.width;
} function getBase() {
return Rectangle;
} class Square extends getBase() {
constructor(length) {
super(length, length);
}
} let x = new Square(10); console.log(x.getArea());

任意表达式都能在extends关键字后使用,但并非所有表达式的结果都是一个有效的类。

  1. null
  2. 生成器函数

继承内置对象

在ES6类的继承中,this的值会先被基类创建,随后才被派生类的构造器所修改。结果是this初始就拥有作为基类的内置对象的所有功能,并能正确的接收与之关联的所有功能。

class MyArray extends Array {

}

let arr = new MyArray();
arr[0] = 'red';
console.log(arr.length); arr.length = 0;
console.log(arr[0]);

Symbol.species属性

继承内置对象的一个有趣方面是:任意能返回内置对象实例的方法,在派生类上却会自动返回派生类的实例。

class MyArray extends Array {

}

let arr = new MyArray(1, 2, 3);
let subitems = arr.slice(1, 2);
console.log(subitems);

深入理解ES6之——JS类的相关知识的更多相关文章

  1. js作用域的相关知识

    众所周知,在ES6之前,JavaScript是没有块级作用域的,如下图所示: 学过其他语言的同学肯定有点诧异,为什么会这样呢?因为js还是不同于其他语言的,在ES5中,只有全局作用域和函数作用域,并没 ...

  2. js预解析相关知识总结以及一些好玩的面试题

    js预解析的题像在做智力题一样有意思~ 预解析 预解析:在解释这行代码之前发生的事情——变量的声明提前了,函数的声明提前 console.log(num) ——未定义Num,结果是报错 var num ...

  3. python类的相关知识第一部分

    一.类的相关概念 (1).什么是类 具有同种属性的对象称为类,是个抽象的概念.比如说:汽车.人.狗.神: (2).什么是对象或实例 日常生活中的所有东西都是对象,是类的实例化.比如说:推土车是汽车的实 ...

  4. JS的数组相关知识

    创建数组方法一: var a1=new Array(5); console.log(a1.length); console.log(a1); //[] ,数组是空的 var a2=new Array( ...

  5. python类的相关知识第二部分

    类的继承.多态.封装 一.类的继承 1.应用场景: 类大部分功能相同,大类包含小类的情况 例如: 动物类 共性:都要吃喝拉撒.都有头有脚 特性: 猫类.走了很轻,叫声特别,喜欢白天睡觉 狗类.的叫声很 ...

  6. js鼠标事件相关知识

    1.mousedown->mouseup依次触发后相当于click事件 2.除了mouseenter和mouseleave外,其它的鼠标事件都是冒泡的 3.mouseover和mouseout事 ...

  7. Java常用类Date相关知识

    Date:类 Date 表示特定的瞬间,精确到毫秒. 在 JDK 1.1 之前,类 Date 有两个其他的函数.它允许把日期解释为年.月.日.小时.分钟和秒值.它也允许格式化和解析日期字符串. Dat ...

  8. 【读书笔记】【深入理解ES6】#9-JavaScript中的类

    大多数面向对象的编程语言都支持类和类继承的特性,而JavaScript却不支持这些特性,只能通过其他方法定义并关联多个相似的对象.这个状态一直从ECMAScript 1持续到ECMAScript 5. ...

  9. es6中class类的全方面理解(一)

    传统的javascript中只有对象,没有类的概念.它是基于原型的面向对象语言.原型对象特点就是将自身的属性共享给新对象.这样的写法相对于其它传统面向对象语言来讲,很有一种独树一帜的感脚!非常容易让人 ...

随机推荐

  1. admin的基础配置

    admin自定义配置 一.admin.py 我们知道在models.py文件中创建的数据表,一方面我们可以通过视图函数对其进行增删改查,一方面我们也可以通过admin进行,通常我们是通过admin的前 ...

  2. Linux入门(11)——Ubuntu16.04安装texlive2016并配置texmaker和sublime text3

    安装过程中需要调用 Perl 的模块 Digest::MD5 来检测 ISO 文件的完整性:升级过程中界面需要调用 Perl 的模块 Tk: sudo apt-get install libdiges ...

  3. 01Vue数据双向绑定

    Vue作为前端MV*架构,Vue.js (读音 /vjuː/,类似于 view) 是一套构建用户界面的渐进式框架.与其他重量级框架不同的是,Vue 采用自底向上增量开发的设计. Vue 的核心库只关注 ...

  4. Pyhton编程(四)之基本数据类型-字符串详解

    一:字符串是什么? 字符串是Python最常用的一种数据类型,虽然看似简单,但能够以不同的方式来使用它们. 字符串就是一系列的字符,在Python中,用引号括起来的都是字符串,其中的引号可以是单引号, ...

  5. jQuery Ajax跨域问题简易解决方案

    场景:由于业务需求,需要在一个页面显示另外一个页面,并且右键查看源代码看不到一条链接. 实现方式:使用iframe来显示这个首页,至于首页的地址则使用jQuery Ajax来获取.html代码如下: ...

  6. 解析 .Net Core 注入 (3) 创建对象

    回顾 通过前两节的学习,我们知道 IServiceCollection 以元数据(ServiceDescriptor)的形式存放着用户注册的服务,它的 IServiceCollection 的拓展方法 ...

  7. MySql中利用insert into select 准备数据uuid主键冲突

    MYSQL 中表1需要准备大量数据,内容主要取自表2,id必须为32位uuid (项目所有表都是这样,没办法), 准备这样插入: INSERT INTO TBL_ONE (ID, SOID, SNAM ...

  8. Fastify 系列教程三 (验证、序列化和生命周期)

    Fastify 系列教程: Fastify 系列教程一 (路由和日志) Fastify 系列教程二 (中间件.钩子函数和装饰器) Fastify 系列教程三 (验证.序列化和生命周期) 验证 Fast ...

  9. 深入理解 Linux 的 RCU 机制

    欢迎大家前往腾讯云社区,获取更多腾讯海量技术实践干货哦~ 作者:梁康 RCU(Read-Copy Update),是 Linux 中比较重要的一种同步机制.顾名思义就是"读,拷贝更新&quo ...

  10. [C#]使用ILMerge将源DLL合并到目标EXE(.NET4.6.2)

    本文为原创文章,如转载,请在网页明显位置标明原文名称.作者及网址,谢谢! 本文主要是使用微软的ILMerge工具将源DLL合并到目标EXE,因此,需要下载以下工具: https://www.micro ...