【问题标题】:How to use private class fields in nodejs 12?如何在 nodejs 12 中使用私有类字段?
【发布时间】:2020-01-02 15:01:49
【问题描述】:

在当前版本的 nodejs 12.x.x,我们可以通过#some_varible 表示法声明private 类字段。 # 表示法将使该特定类的变量私有字段。

class Foo {
  #some_varible = 10;
}

我有以下问题:

  • 如何使用这个变量?
  • 在生产应用程序中使用此类符号来声明私有类字段是否安全?
  • 当某些类字段被声明为私有时,它如何在后台(在 nodejs 中)工作?

【问题讨论】:

标签: javascript node.js private class-fields


【解决方案1】:

如何使用这个变量?

您可以将其用作访问器属性的支持字段:

class Foo {
  #some_variable = 10;
  
  get some_variable () {
    console.log('getting some_variable');
    return this.#some_variable;
  }
  
  set some_variable (value) {
    console.log('setting some_variable');
    this.#some_variable = value;
  }
}

let bar = new Foo();

console.log(bar.some_variable);
console.log(bar.some_variable = 42);

或者您可以像普通属性一样使用它,主要区别在于它只能从class { ... } 范围内访问,因为它是私有的,并且该属性以# 为前缀。

还要注意this.#some_variablethis['#some_variable']相同,后者指的是字符串属性,而不是私有字段。

在生产应用程序中使用此类符号来声明私有类字段是否安全?

答案是视情况而定

使用客户端 JavaScript,您无法控制执行环境,但您可以使用 Babel plugins 将新语言功能(例如私有类字段)转换为仅依赖于旧语言规范中的语法的功能等效项。

在 Node.js 中,您确实可以控制执行环境,因此要使用新的语言功能,只需确保您使用的是 a version that supports the syntax

当某些类字段被声明为私有时,它如何在后台(在 nodejs 中)工作?

V8 实现

私有字段使用块范围的symbols,它被反射方法Object.getOwnPropertySymbols()列入黑名单。

模拟此实现的转换类可能如下所示:

let Foo;

{
  const some_variable = Symbol('#some_variable');

  {
    const getOwnPropertySymbolsImpl = Object.getOwnPropertySymbols;
    Object.getOwnPropertySymbols = function getOwnPropertySymbols () {
      return getOwnPropertySymbolsImpl.apply(this, arguments)
        .filter(symbol => symbol !== some_variable);
    };
  }

  Foo = class Foo {
    [some_variable] = 10;
  };
}

let bar = new Foo();
console.log(bar);

这会产生与本机实现非常相似的输出:


Babel 实现

@babel/plugin-proposal-class-properties

私有字段存储为WeakMap,其中每个条目是一个类实例键及其各自的私有字段property descriptor 值。

以下是 Babel 如何转换我们一直使用的类作为示例:

class Foo {
  constructor() {
    _some_variable.set(this, {
      writable: true,
      value: 10
    });
  }
}

var _some_variable = new WeakMap();

请注意,虽然_some_variable 在这里不是块作用域,但 Babel 将确保转换后的输出中的名称不会与该作用域中的任何可见名称冲突,因此无法从预编译源中访问弱映射.

【讨论】:

  • 哇,谢谢!这确实有助于理解这个概念。
猜你喜欢
  • 2019-12-14
  • 2011-05-09
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2020-04-25
  • 2012-01-23
  • 1970-01-01
相关资源
最近更新 更多