static

类(class)通过 static 关键字定义静态方法。不能在类的实例上调用静态方法,而应该通过类本身调用。这些通常是实用程序方法,例如创建或克隆对象的功能。

语法

static methodName(){ .....  }

描述
静态方法调用直接在类上进行,不能在类的实例上调用。静态方法通常用于创建实用程序函数

调用静态方法

静态方法调用同一个类中的其他静态方法

静态方法调用同一个类中的其他静态方法,可使用this关键字。

class StaticMethodCall {
    static staticMethod() {
        return 'Static method has been called';
    }
    //可以使用 this调用另一个静态方法
    static anotherStaticMethod() {
        return this.staticMethod() + ' from another static method';
    }
}
StaticMethodCall.staticMethod();
// 'Static method has been called'
 
StaticMethodCall.anotherStaticMethod();
// 'Static method has been called from another static method'

类的构造函数和其他方法(非静态方法中)

非静态方法中,不能直接使用 this关键字来访问静态方法。而是要用类名来调用:CLASSNAME.STATIC_METHOD_NAME() ,或者用构造函数的属性来调用该方法: this.constructor.STATIC_METHOD_NAME().

class StaticMethodCall {
    constructor() {
        console.log(StaticMethodCall.staticMethod());
        // 'static method has been called.'
        console.log(this.constructor.staticMethod());
        // 'static method has been called.'
    }
    static staticMethod() {
        return 'static method has been called.';
    }
}

public

对象的成员都是public成员。任何对象都可以访问,修改,删除这些成员或添加新成员。主要有两种方式来在一个新对象里放置成员:

在构造函数里

这种技术通常用来初始化public实例变量。构造函数的“this”变量用来给对象添加成员。

functin Container(param) {
    this.member = param; 
}

这样,如果我们构造一个新对象var myContainer = new Container('abc'),则myContainer.member为'abc'。

在prototype里

这种技术通常用来添加public方法。当寻找一个成员并且它不在对象本身里时,则从对象的构造函数的prototype成员里找。
prototype机制用来做继承。为了添加一个方法到构造函数创建的所有对象里,只需添加到构造函数的prototype:

Container.prototype.stamp = function (string) {
    return this.member + string;
}

private

private成员由构造函数产生。普通的var变量和构造函数的参数都称为private成员。

function Container(param) {
 
    this.member = param;
     //私有属性
    var secret = 3;
     //私有属性
    var that = this;
 
}

该构造函数创建了3个private实例变量: param,secret和that。它们被添加到对象中,但是不能被外部访问,也不能被该对象自己的 public方法访问。它们只能由private方法访问。private方法是构造函数的内部方法。

function People () {
    this.name = "Yorhom";
    var age = 16;
    //实例中的方法
    this.getName = function () {
         return this.name
    };
    //实例方法 使用闭包的原理实现 私有属性的访问
    this.getAge = function () {
        return age;
    };
}
 
var yorhom = new People();
// undefined
alert(yorhom.age);
// 16
alert(yorhom.getAge())

Class Fields 提案的另一个内容时是 “private fields (私有属性)” 。 有时,当您构建一个类时,您希望拥有不暴露给外界的私有值。 从历史上看, JavaScript 缺乏真正私有值 的能力,所以我们通过约定,用下划线标记它们。

class Car {
   //私有属性 
  _milesDriven = 0
  drive(distance) {
    this._milesDriven += distance
  }
  getMilesDriven() {
    return this._milesDriven
  }
}

在上面的示例中,我们依靠 Car class(类)的实例通过调用 getMilesDriven 方法来获取汽车的里程数。但是,因为没有什么能使 _milesDriven成为私有的,所以任何实例都可以访问它。

const tesla = new Car()
tesla.drive(10)
console.log(tesla._milesDriven)//undefined

有个奇特的(hacky)方法,就是使用 WeakMaps 可以解决这个问题,但如果存在更简单的解决方案,那将会很好。 同样,Class Fields 提案正在拯救我们。 根据提议,您可以使用 # 创建私有字段。 是的,你没有看错, # 。 我们来看看它对我们的代码有什么影响,

class Car {
  #milesDriven = 0
  drive(distance) {
    this.#milesDriven += distance
  }
  getMilesDriven() {
    return this.#milesDriven
  }
}

我们可以用速记语法更进一步简化

class Car {
  #milesDriven = 0
  drive(distance) {
    #milesDriven += distance
  }
  getMilesDriven() {
    return #milesDriven
  }
}
 
const tesla = new Car()
tesla.drive(10)
tesla.getMilesDriven() // 10
tesla.#milesDriven // Invalid

protected

protected可以修饰数据成员,构造方法,方法成员,不能修饰类(此处指外 部类,不考虑内部类)。被protected修饰的成员,能在定义它们的类中,同包 的类中被调用。如果有不同包的类想调用它们,那么这个类必须是定义它们的类 的子类。

//module foo:
 
class Foo {
    constructor() {
        this[Foo.PROPERTY] = 'hello';
    }
 
    test() {
      console.log(this[Foo.PROPERTY]);
    }
}
 
Foo.PROPERTY = Symbol();
 
export default Foo;
//module bar:
 
import Foo from '(module foo)';
 
class Bar extends Foo {
    test2() {
      console.log(this[Bar.PROPERTY]);
    }
}
 
export default Bar;
//module main:
 
import Bar from '(module bar)';
 
new Bar.test2();

总结:

关键字 类本身 类的方法 类的实例 子类 子类方法 子类的实例
static + - - + - -
public - + + - + +
private - + - - - -
protected - + - - + -

————————————————
【参考文献】https://www.html.cn/archives/...
【原文链接】https://blog.csdn.net/qq_3804...

静态属性和方法的定义

静态属性和方法以及静态类在js中的定义非常简单,先来看静态类:

var StaticClass = {};

这么写不是在定义一个Object吗?是的,不错,不过js中的静态类也是可以这样定义的。如果要添加静态类中的方法和属性,就可以这么写:

var StaticClass = {
    id : 5,
    sayHello : function () {
        alert("Hello"); 
    }
};

如果是要向类中添加静态属性或者方法,可以采用这种写法:

function People () {
    this.name = "Yorhom";
}

People.prototype.getName = function () {
    return this.name;
};

People.TYPE = "people";
People.sayHello = function () {
    alert("Hello");
};

JavaScript Set/Get访问器

什么是set/get访问器呢?如果你熟悉python,那么你可以理解为@property和@xxx.setter,但是简陋的js里也有?当然有,只不过是ES5的标准,可以采用这种写法:

Object.defineProperty(this, "name", {
    get : funtion () {
        return name;
    },

    set : function (v) {
        name = v;
    }
});

具体有什么用呢?大致就是this.name属性在被获取的时候调用get访问器,在被更改值的时候调用set。

如何“歪门邪道”地做到禁止访问私有和保护属性?

在常规开发下,只能通过闭包来阻止某变量的访问。可是如果你使用了prototype,那么闭包这条路就走不通了。在这种情况下,我们的Object.defineProperty就出场了。我们知道,通过这个函数可以设定获取属性时返回的值,也可以设定更改属性时设置的值。有了这个函数,我们可以随时跟踪到某个属性是不是在被获取,或者是不是在被更改。我们还需要一个开关,我们在类内部的方法调用时,把这个开关打开,表明是在内部运行,方法调用结束后将开关关闭,表明回到外部运行状态。有了这两个状态,我们就可以跟踪private和protected属性和方法了,一旦他们在开关关闭的时候被使用,就终止这个属性或方法的获取或设置。
于是乎,大难题就快解决了。

————————————————
原文链接:https://blog.csdn.net/yorhomw...


一吃三大碗
130 声望9 粉丝

“唯有深入,方能浅出”