3
熟悉面向对象编程的都知道,面向对象编程最重要的原则之一 - 从外部接口划分内部接口。也就是说,针对某一类事物,我们其实并不是那么在乎其内部究竟是怎样去实现的,只关心怎样使用而已。

为了理解这点,让我们先来看看现实生活中的列子。通常,我们使用的设备非常复杂。但是从外部接口界定内部接口允许使用它们没有什么问题。列如一辆汽车。从外面看主要有:轮子、车身、车顶、方向盘等。
汽车外部组成

但是,内部...
汽车内部结构图

有许多的细节,但是我们并不用去知道这些细节,也可以很好地开车车。

汽车非常可靠,不是吗?我们可以使用很多年,只有在出现问题时才能使用它 - 进行维修。汽车的可靠性和使用的简单性在于隐藏内部细节。

如果我们从汽车上取下发动机,那么使用它将会复杂得多(安装在哪?),并且危险(它可以电击)。

所以,面向对象编程就类似于汽车一样。

内部和外部接口

在面向对象的编程中,属性和方法可以分为两组:

  • 内部接口-方法和属性,可以从类的其他方法访问,但不能从外部访问
  • 外部接口-方法和属性,也可以从外部访问

如果我们继续与汽车进行类比 - 内部隐藏的部分:发动机、变速器、半轴等 - 是其内部接口。对于对象的运行,内部接口是非常有用的,其细节互相使用。例如,弹性元件连接到减震器。

但是从外面看,汽车被外层车壳保护着,所以没有人可以接触到。细节隐藏且无法访问。我们可以通过外部接口使用它的功能。所以,我们在使用一个对象的时候,并不关心它内部是怎样工作的。

在JavaScript中,有两种类型的对象字段(属性和方法):

  • 公共的:随处都可访问,它们包含外部接口,我们在开发中一直常用的也就是公共的属性和方法了
  • 私有的:仅在类的内部可访问,主要用于内部接口

在许多其他语言中,还存在“受保护”字段:只能从类内部访问和扩展它们。它们对内部接口也很有用。它们在某种意义上比私有更广泛,因为我们通常希望通过继承类来获取和访问它们。

受保护的字段不是在JavaScript语言级别上实现的,但实际上它们非常方便,我们也可以模拟地去实现它们。现在我们用JavaScript来制作一台具有这些类型属性的汽车。

class MBWCar{
    oilAmount = 0; // the amount of oil inside
    constructor(power){
        this.power = power
        alert( `Created a mbw-car, power: ${power}` );
    }
}
// create the mbw car
let mbwCar = new MBWCar(100)

// oil water
mbwCar.oilAmount = 200

从上面的代码可以看出oilAmountpower这两个属性是公共的,我们可以在外部轻易地设置以及获取它们。

让我们将oilAmount属性更改为protected以对其进行更多控制。例如,我们不希望任何人将其设置为零以下。

受保护的属性通常以下划线_为前缀

这不是在语言层面强制去执行,但咱们程序员之间有一个众所周知的惯例,即不应该从外部访问这些属性和方法。

class MBWCar{
    _oilAmount = 0;

    constructor(power){
        this._power = power

    }

    set oilAmount(value){
        if (value < 0) throw new Error("Negative oil");
        this._oilAmount = value
    }

    get oilAmount(){
        return this._oilAmount
    }
}
// create the mbw car
let mbwCar = new MBWCar(100)

// oil water
mbwCar.oilAmount = -10 //Error Negative oil

现在访问受到控制,因此将油量设置为零以下将失败。

那么我们可以把power属性设置为只读属性,这在一些程序开发中也有类似的需要,某些属性只读不可更改其值。

class MBWCar{
    constructor(power){
        this._power = power

    }
    get power(){
        return this._power
    }
}
// create the mbw car
let mbwCar = new MBWCar(100)
alert(`Power is: ${mbwCar.power}W`); // Power is: 100W

mbwCar.power = 25 // Error (no setter)

getter/setter方法

 class Car{
    _oilMount = 0;
    setOilMount(value){
        if(value<0) throw Error('Negative oil')
        this._oilMount = value
    }
    getOilMount() {
        return this._oilMount;
    }
}
let mbw = new Car()
mbw.setOilMount(100);
alert(mbw.getOilMount());

受保护的属性是可以继承的

如果我们继承类MBWCar扩展Car,那么没有什么能阻止我们从新类的方法中访问this._oilMount 或this._power属性。
所以受保护的属性是可以继承的,不像接下来我们所说的私有属性。

私有字段

这是JavaScript后面新增的一个针对类属性的语法。JavaScript引擎不支持,或者部分支持,需要进行polyfilling
在JavaScript的提议中,有一个已完成的标准,为私有属性和方法提供语言级支持。

私有的私有字段与#开头,仅在类的内部可进行访问.

  class Car{
    #oilLiMit = 100;

    #checkOil(value){
        if(value<0) throw Error(`Negative oil`)
        if (value > this.#waterLimit) throw new Error("Too much oil");
    }
}

let car = new Car();

// can't access privates from outside of the class
car.#checkOil(); // Error
car.#waterLimit = 1000; // Error

class BWMCar extends Car{
    method() {
        alert( this.#oilLiMit ); // Error: can only access from Car
    }
}

私有字段不能够通过this[name]去访问

 class User{
   sayHi(){
       let filedName = "Darkcod"
       alert(`Hello,${this.filedName}`) //Hello,undefined
   }
}

 new User().sayHi()
 

总结:

  1. 面向对象编程最重要的原则之一 - 从外部接口划分内部接口,这就涉及到属性和方法的可访问范围度
  2. 在面向对象编程中,可通过private、protected、public来对类的属性和方法进行限制,例如你从你父亲那里继承了一些属性,但你父亲其他属性不像被你继承到等。
  3. 在JavaScript中,受保护的属性和方法名以_开头,私有的属性和方法名以#开头
  4. 面向对象编程的一个较大的好处之一是我们不必理解其内部实现,依然可以很好地去进行编程开发,例如:一个U盘,我们想要将电脑中的某些文件拷贝到它那里进行存储,这时候我们并不关心U盘的内部是这样形成的,我们只知道通过USB口插入即可完成拷贝的工作

前端扫地僧
2.5k 声望1.2k 粉丝