详解Class

  Classs是es6提供的类,相当于es5的构造函数。

  写法: 

class Foo {
    constructor () { // new 的时候会调用该方法,可以通过return改变构造函数的返回值
        return Object.create(null)
    }
}
let foo = new Foo()
//console.log(foo instanceof Foo) // false Foo为null

// 类的实例对象
class People {
    constructor () {
        this.sex = 'man';
        this.name = 'liming';
    }
    changeName (value) {
        this.name = value;
    }
}

let person = new People();
console.log(person.hasOwnProperty('name')) // true hasOwnProperty 查看属性是否挂在对象本身
console.log(person.hasOwnProperty('changeName')) // false
console.log(person.__proto__.hasOwnProperty('changeName')) // true _proto_ 访问原型对象

  由上面的代码可以看出,Class与es5的构造函数大体相同。都使通过new关键字创建实例对象,Class中不在constructor中的方法均属于挂在原型上的方法。相当于functionname.prototype.funname1.

  Class不存在变量提升,使用Class之前,必须保证已经定义。

  Class的私有方法和私有属性 

let baz = Symbol('baz')
class Animal {
    changeName (name) {
        changeName.bind(this, name) // 私有 将方法移到外部创建
    }
    _setName () { // 私有 格式区分
    }
    [baz](baz){} // 利用Symbol创建私有方法
}

function changeName(val) {
    return this.name = val;
}

  创建私有方法有三种:

  • 通过方法名区分_。
  • 通过Symbol对象。
  • 将私有方法放到类的外部。

  Class中的this,


class A {
constructor(){
this.x = 1;
this.y = 2;
}
aa(){
return this.x
}
}

let a = new A();
let {aa} = a;
console.log(aa()) // Cannot read property 'x' of undefined
console.log(a) // { x: 1, y: 2 }

  在上面的代码中a实例对象时可以调用a.aa()方法的,但是用解构赋值之后,调用aa()方法报错,输出实例对象a没有看到aa方法。

  理解原因,我们需要知道new的过程:

  1. 创建空对象
  2. 将构造函数的方法和属性挂在空对象上
  3. 将this指向空对象

  上面我们已经说过aa方法是挂在原型上的方法,并不是属于类A的。所以调用aa()会报错。下面我们将aa()方法挂在类的构造函数上面。

  

class A {
    constructor(){
        this.x = 1;
        this.y = 2;
        this.aa = this.aa.bind(this); // 这句可解决下面的问题
    }
    aa(){
        return this.x
    }
}

let a = new A();
let {aa} = a;
console.log(aa()) // Cannot read property 'x' of undefined 原因解构赋值值读取对象本身的属性,不读取原型链
// 可以将 aa 函数放到constructor函数中 或者将方法重新绑定到this上
console.log(a) //  { x: 1, y: 2, aa: [Function: bound aa] }

  Class的取值函数和存值函数。 

class B {
    get prop () {
        return 'getter'
    }
    set prop (val) {
        console.log('shezhi' + val)
    }
}

var b = new B()
console.log(b.prop)
b.prop = 3;
console.log(b.prop)

  Class 的静态方法(不希望被实例对象继承的方法可以被类继承)。

 

class C{
    static getname () { // 关键字static
        return this.name; // 此this指向的是类而不是实例
    }
}

console.log(C.getname())

let c = new C()
//c.getname() // c.getname is not a function

class D extends C {

}
console.log(D.getname()) // D

  Class的静态属性。

  

/*
* class 的静态属性和实例属性
* 静态属性指的是挂在类上的属性不是实例上的(this上的)可以直接通过类访问到的
* 实例属性是实例对象上的属性
* */
class E{
    //static num = 3; // 静态属性
    //num = 3; // 实例属性
    constructor () {
        this.a = 4; // 实例属性
    }
}
E.c = 5; // c 静态属性
console.log(new E().a)

   Class继承。

  

/*class 继承
es5 通过修改原型链继承
es6 通过extends实现继承*/

// es5
function ParentG() {
    this.x = 1;
    this.y = 2;
    this.add = function () {
        console.log(1)
    }
}

function childG() {}
childG.prototype = new ParentG(); // 通过改变原型对象实现继承
console.log(new childG().add())

// es6
class ParentH {
    constructor () {
        this.x = 21;
    }
    add () {
        console.log(2)
    }
}

class childH extends ParentH { // extends 实现继承
}
console.log(new childH().x)

// super 获取父类的this 必须在子类的构造函数中调用super() super内部的this指向子类的实例
class parentJ {
    constructor () {
        this.a = 1;
        this.b = 2;
    }
    add () {
        return 'add'
    }
}

class childJ extends parentJ{
    constructor (a, b) {
        super(a, b); // 继承的子类没有this必须调用super
        this.c = 3;
    }
    add () {
        console.log('子类' + super.add())
    }
}
console.log(new childJ().add())

/*
* 区别 es5的继承只是改写子类的原型对象本质上子类和父类是独立的连个对象
* es6的继承只是对父类的改写或补充,子类没有this
* */

// Object.getPrototypeOf() 从子类上获取父类
let parentClass = Object.getPrototypeOf(childH)
console.log(parentClass) // [Function: ParentH]

// 类的prototype 和 __proto__ 下面我们将这两个属性成为a和b
// es5 中每个对象都有a和b属性,对象的b属性指向创建对象的构造函数的a属性即构造函数的原型对象

function L() {}
let l = new L();
l.__proto__ == L.prototype;

// es6 的Class 也具有prototype和__proto__属性
// 子类的__proto__ 属性表示构造函数的继承总是指向父类
// 子类的prototype属性的__proto__的属性表示方法的继承,总是指向父类的prototype
class ParentY {}
class ChildY extends ParentY {}
console.log(ChildY.__proto__ === ParentY) // true
console.log(ChildY.prototype.__proto__ === ParentY.prototype) // true

// 继承的原理
Object.setPrototypeOf(ChildY.prototype, ParentY.prototype) // 子类的实例继承父类的实例
Object.setPrototypeOf(ChildY, ParentY) // 子类继承父类的静态属性

// setPrototypeOf 的实现方法
Object.setPrototypeOf = function (obj, proto) {
    obj.__proto__ = proto;
    return obj;
}

// 子类作为对象时,子类的原型父类,作为构造函数,子类的原型对象时父类原型对象的实例(和es5实现继承一样)

 

 

posted @ 2018-04-12 18:02  木子青牛  阅读(273)  评论(0编辑  收藏  举报