首頁  >  文章  >  web前端  >  class在es6中本質是什麼

class在es6中本質是什麼

青灯夜游
青灯夜游原創
2022-10-28 19:01:341221瀏覽

class在es6中本質是函數(建構器),使用的時候,也是直接對類別使用new指令,跟建構子的用法一致;class可以看成一個語法糖,讓物件原型的寫法更清晰、更像物件導向程式設計的語法。用class定義類別的方法「class Person{//類別宣告}」或「const Person=class{//類別表達式}」。

class在es6中本質是什麼

本教學操作環境:windows7系統、ECMAScript 6版、Dell G3電腦。

es6的Class類別詳解

class基本語法

JavaScript 語言中,產生實例物件的傳統方法是透過建構函數和原型的組合模式.ES6 提供了更接近傳統語言(java)的寫法,引入了Class(類別)這個概念,作為物件的模板。透過class關鍵字,可以定義類別。

class point{ 
   
	constructor(x,y){ 
   
        this.x=x;
        this.y=y;
    }
    play(){ 
   
        console.log("我会玩");
    }
}

ES6 的class可以看作只是一個語法糖,它的絕大部分功能,ES5 都可以做到,新的class寫法只是讓物件原型的寫法更加清晰、更像物件導向程式設計的文法而已。

註:「語法糖」:是由英國電腦科學家彼得·約翰·蘭達(Peter J. Landin)發明的術語,指電腦語言中添加的某種語法,這種語法對語言的功能並沒有影響,但更方便程式設計師使用。

class在ES6本質就是函數(建構器)

class在es6中本質是什麼

#class中的原型方法全部被加入到了parent的原型對象上

class在es6中本質是什麼

ES6 的class與ES5寫法的幾個核心注意點: ES5 的建構子Point,對應 ES6 的Point類別的建構方法。 類別的所有方法都定義在類別的prototype屬性上面。 定義「類別」的方法的時候,前面不需要加上function這個關鍵字,直接把函數定義放進去了就可以了 方法之間不需要逗號分隔,加了會報錯 ES6的class使用方法與ES5的建構子一模一樣

//类的所有方法都定义在类的prototype属性上面。
class piont{ 
   
    constructor(){ 
   
		//
    }
    play(){ 
   
        
    }
}
//上述代码等价于
point.prototype={ 
   
    constructor() { 
   },
    play(){ 
   };
}

//在类的实例上面调用方法,其实就是调用原型上的方法。
class Ba{ 
   
	//
}
let b=new Ba();
b.constructor===Ba.prototype.constructor//true

另外:ES5 的建構子Point,對應 ES6 的Point類別的建構方法。

由於類別的方法都定義在prototype物件上面,所以類別的新方法可以加入prototype物件上面。 Object.assign方法可以很方便地一次向類別新增多個方法。

class ponit{ 
   
    constructor(){ 
   
        
    }
}
Object.assign(Point.prototype,{ 
   
	play(){ 
   };
})
//Class直接定义的方法之间不需要逗号分隔,加了会报错. 但是这里是Object.assign的方法格式, 这里面需要往Point.prototype里面添加的方法就需要符合对象的默认格式

類別的內部所有定義的方法,都是不可列舉的(non-enumerable)。透過Object.assign方法往類別的原型上新增的方法,constructor不可枚舉, 其他的可以枚舉

Class的基本語法之constructor

constructor方法是類別的預設方法,透過new指令產生物件實例時,自動呼叫該方法。一個類別必須有constructor方法,如果沒有明確定義,一個空的constructor方法會被預設為添加。

constructor方法預設回傳實例物件(即this),完全可以指定傳回另外一個物件 (得是在創造class時就定義設定的, 在創造完class後,透過Object.assign的方式是沒法改變建構子的回傳值的).

Class的基本語法之類別的調用方式

類別必須使用new調用,否則會報錯。這是它跟普通建構函式 ( 普通建構函式完全可以當做普通函式使用 ) 的一個主要區別,後者不用new也可以執行。

class在es6中本質是什麼

#Class的基本語法之getter和setter

#與ES5 一樣,在「類別」的內部可以使用get和set關鍵字,對某個屬性設定存值函數和取值函數,攔截該屬性的存取行為。

  class demo{ 
   
            constructor(age){ 
   
                this.age=agie;
                this._age=age;
            }
            get age(){ 
   
                return this._age;
            }
            set age(value){ 
   
                this._age=value;
                console.log("年龄"+value);
            }
        }
        let kevin=new demo(9);
        kevin.age=18;
        console.log(kevin.age);
Class的基本语法之类的属性名

class在es6中本質是什麼

上面代码中,Square类的方法名getArea,是从表达式得到的。

Class的基本语法的特别注意点

(1)严格模式

类和模块的内部,默认就是严格模式,所以不需要使用use strict指定运行模式。只要你的代码写在类或模块之中,就只有严格模式可用。考虑到未来所有的代码,其实都是运行在模块之中,所以 ES6 实际上把整个语言升级到了严格模式。

(2)不存在提升

new foo();
class foo{};

上面代码中,Foo类使用在前,定义在后,这样会报错,因为 ES6 不会把类的声明提升到代码头部。

(3)name 属性

class point{
}
point.name//point

由于本质上,ES6 的类只是 ES5 的构造函数的一层包装,所以函数的许多特性都被Class继承,包括name属性。

(4)this 的指向

类的方法内部如果含有this,它默认指向类的实例。但是,必须非常小心,一旦单独使用该方法,很可能报错。

class在es6中本質是什麼

printName方法中的this,默认指向Logger类的实例。但是,如果将这个方法提取出来单独使用,this会指向该方法运行时所在的环境(由于 class 内部是严格模式,所以 this 实际指向的是undefined),从而导致找不到print方法而报错。

解决办法:

一个比较简单的解决方法是,在构造方法中绑定this,这样就不会找不到print方法了。

另一种解决方法是使用箭头函数。箭头函数位于构造函数内部,它的定义生效的时候,是在构造函数执行的时候。这时,箭头函数所在的运行环境,肯定是实例对象,所以this会总是指向实例对象。

class Logger{ 
   
    constructor(){ 
   
        this.printName=this.printName.bind(this);
        //但是请注意bind之后返回的函数里面的this就永久锁死了问题:!!! !!! 坚决别用 
    }
}
//箭头函数
class Obj{ 
   
    constructor(){ 
   
        this.getThis=()=>this;
    }
}
let o=new Obj();
o.getThis()===o//true

Class的静态属性和方法

类相当于实例的原型,所有在类中定义的方法,都会被实例继承。如果在一个方法前,加上static关键字,就表示该方法不会被实例继承,而是直接通过类来调用,这就称为“静态方法”。

 class Person{ 
   
            static sum=0;
            constructor(){ 
   
                this.add();
            }
            add(){ 
   
                Person.sum++;
            }
        }
        let kaiwen=new Person();
        console.log("当前的聊天室人数为:"+Person.sum);
        //作用:当没有实例化的时候,我们可以通过静态的属性和方法去获取一些信息
 // 注意,如果静态方法包含this关键字,这个this指的是类,而不是实例。静态方法可以与非静态方法重名。

父类的静态方法,可以被子类继承静态方法也是可以从super对象上调用的。

 class Person{ 
   
            constructor(name){ 
   
                this.name=name;
                this.sex="男";
            }
        }
        class Student extends Person{ 
   
            constructor(name,age){ 
   
                super(name);
                this.age=age;
            }
        }
        let s=new Student("张三",11);
        console.log(s.name);
        console.log(s.age);
        console.log(s.sex);

Class的私有方法和私有属性

私有方法和私有属性:是只能在类的内部访问的方法和属性,外部不能访问。 这是常见需求,有利于代码的封装,但 ES6 不提供,只能通过变通方法模拟实现。

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

下面代码中的写法不仅可以写私有属性,还可以用来写私有方法

 class Cat{ 
   
            #eyes="眼睛";
            static pai(){ 
   
                console.log("凯文");
            }
            say(){ 
   
                Cat.pai();
                console.log("猫有一双大大的"+this.#eyes);
            }
        }
        let kate=new Cat();
        kate.say();

私有属性也可以设置 getter 和 setter 方法。

私有属性不限于从this引用,只要是在类的内部,实例也可以引用私有属性。

构造函数的新属性

ES6 为new命令引入了一个new.target属性,该属性一般用在构造函数之中,返回new命令作用于的那个构造函数。如果构造函数不是通过new命令调用的,new.target会返回undefined,因此这个属性可以用来确定构造函数是怎么调用的。

私有属性也可以设置 getter 和 setter 方法。

私有屬性不限於從this引用,只要是在類別的內部,實例也可以引用私有屬性。

建構子的新屬性

ES6 為new指令引進了一個new.target屬性,該屬性一般用在建構函式之中,回傳new指令作用於的那個構造函數。如果建構子不是透過new指令呼叫的,new.target會回傳undefined,因此這個屬性可以用來決定建構子是怎麼呼叫的。

class在es6中本質是什麼

#【相關推薦:javascript影片教學程式影片

以上是class在es6中本質是什麼的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn