如何使用这个变量?
您可以将其用作访问器属性的支持字段:
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_variable不与this['#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 实现
私有字段存储为WeakMap,其中每个条目是一个类实例键及其各自的私有字段property descriptor 值。
以下是 Babel 如何转换我们一直使用的类作为示例:
class Foo {
constructor() {
_some_variable.set(this, {
writable: true,
value: 10
});
}
}
var _some_variable = new WeakMap();
请注意,虽然_some_variable 在这里不是块作用域,但 Babel 将确保转换后的输出中的名称不会与该作用域中的任何可见名称冲突,因此无法从预编译源中访问弱映射.