这是什么,如何使用,为什么需要?

一边听“Noise Pollution” —— Portugal. The Man,一边阅读本文简直就是享受

JavaScript 标准的第二阶段(Stage 2)加入了类私有字段。它还没有最终确定,但 JavaScript 标准委员会认为这个特性会被开发出来并最终纳入标准(虽然它可能还会改变)

它的语法(当前)看起来像这样:

 class Point {
**#x;**
**#y;** constructor(x, y) {
**this.#x** = x;
**this.#y** = y;
} equals(point) {
return **this.#x** === **point.#x** && **this.#y** === **point.#y**;
}
}

  

这个语法有两个主要部分:

  • 定义私有字段

  • 引用私有字段

定义私有字段

定义私有字段与定义公共字段基本相同:

 class Foo {
**publicFieldName =** 1;
**#privateFieldName =** 2;
}

为了访问私有字段,你需要定义它。如果你不想在定义的时候给它赋值,可以这样:

 class Foo {
**#privateFieldName;**
}

引用私有字段

引用私有字段与访问其它属性类似,只是它有一点特殊的语法。

 class Foo {
publicFieldName = 1;
#privateFieldName = 2; add() {
return this.publicFieldName + **this.#privateFieldName**;
}
}

this.# 也可以简写:

 method() {
**#privateFieldName;**
}

它与下面这段代码等价:

 method() {
**this.#privateFieldName;**
}

引用实例(对象)的私有字段

引用私有字段并不局限于 this。也可以访问类实例(对象)的私有字段:

 class Foo {
#privateValue = 42; static getPrivateValue(**foo**) {
return **foo.#privateValue**;
}
} Foo.getPrivateValue(**new Foo()**); // >> 42

这里,foo 是 Foo 的实例,所以我们允许在类定义中去寻找 #privateValue。

私有方法(很快有会有?)

私有字段这个提案只关注了为类添加私有字段这个问题。提案并没有提到会对类的方法做什么改变,所以私有方法可能会在后续建议中提到,像这样:

 class Foo {
constructor() {
**this.#method();**
} **#method() {**
// ...
**}**
}

这之前,可以将函数赋值给私有字段:

 class Foo {
constructor() {
**this.#method();**
} **#method = () => {**
// ...
**};**
}

封装

使用某个类的实例的时候,不能引用这个类的私有字段。只能在定义这些私有字段的类中引用它们。

 class Foo {
**#bar;** method() {
**this.#bar;** // Works
}
} let foo = new Foo(); **foo.#bar;** // Invalid!

进一步说,作为真正的私有属性,你应该不能检测到私有字段的存在。

为了确保你不能探测到私有字段,我们需要允许拥有同样名称的公共字段。

 class Foo {
bar = 1; // public bar
#bar = 2; // private bar
}

如果私有字段不允许公共字段有同样的名称,你就可以通过尝试写同名属性来探测到私有字段的存在:

 foo.bar = 1; // Error: bar is private! (boom... detected)

或者不报错的版本:

 foo.bar = 1;
console.log(foo.bar); // undefined (boom... detected again)

这种封装同样对子类有效。子类应该可以拥有相同名称的私有字段,而不必担心父类是否已经存某个私有字段是这个名称。

 class Foo {
**#fieldName** = 1;
} class Bar extends Foo {
**fieldName** = 2; // Works!
}

注意:关于封装或“强私有”背后的动机,阅读 FAQ 中这个部分


那么, 为什么用 # 号[译者注:URL 的 Hash标记]?

很多人想知道“为什么不像其它语言那样使用 private 关键字”?

如果使用这样的语法,这里有个示例:

 class Foo {
**private value;** equals(foo) {
return **this.value** === **foo.value**;
}
}

让我们单独看看语法的两个部分。

为什么不使用 private 关键字来声明?

private 关键字在很多不同的语言中用于声明私有字段。

来看看使用这种语法的语言:

 class EnterpriseFoo {
public bar;
private baz; method() {
this.bar;
this.baz;
}
}

在这些语言中,以同样的方式访问私有字段和公共字段。所以它们才会这样定义。

但是在 JavaScript 中,我们不能使用 this.field 来引用私有属性(稍后深入),我们需要一种基于语法的方法来连接它们的关系。这两个地方使用 # 更能清楚的表明引用的是什么。

什么引用需要 # (Hash 标记)?

因为如下原因,我们需要使用 this.#field 而不是 this.field:

  1. 因为 #封装 (参阅上面“封装”一节),我们要能同时访问公共和私有的同名字段。因此访问一个私有字段不是按常规的方式查找。

  2. JavaScript 中的公共字段可以通过 this.field 或 this['field'] 来访问。私有字段不能支持第二种语法(因为它需要是静态的),这样会导致混淆。

  3. 你需要费劲去检查:

来看一段示例代码。

 class Point {
**#x;**
**#y;** constructor(x, y) {
**this.#x** = x;
**this.#y** = y;
} equals(other) {
return **this.#x** === **other.#x** && **this.#y** === **other.#y**;
}
}

注意我们是如何引用 other.#x 和 other.#y 的。为了访问私有字段,我们假设 other 是 Point 类的实例。

因为我们使用了 #Hash标签 语法,它告诉 JavaScript 编译器我们正在从当前类中寻找私有属性。

那么,如果不用 #Hash标签,会发生什么呢?

 equals(otherPoint) {
return this.x === otherPoint.x && this.y === otherPoint.y;
}

现在我们有个问题:我们怎么知道 otherPoint 是什么?

JavaScript 没有静态类型系统,所以 otherPoint 什么都可能是。

有两个原因导致这个问题:

  1. 我们的函数行为依赖于传入的值类型,不同的值类型导致不同的行为:有时候是在访问私有属性,有时候又是在查找公共属性。

  2. 我们每次检查 otherPoint 的类型:

    if (otherPoint instanceof Point && isNotSubClass(otherPoint, Point)) { return getPrivate(otherPoint, 'foo'); } else { return otherPoint.foo; }

更糟糕的是,如果我们引用一个私有属性的话,就不得不在类中对每次属性访问进行这样的检查。

访问属性已经非常慢了,我们绝不想让它变得更慢。


TL;DR: 我们需要对私有属性使用 #Hash标记,因为不正常的使用标准属性访问机制会导致意外行为,并造成巨大的性能问题。


将私有属性添加到语言中是件非常好的事情。感谢努力工作在 TC39 勤劳的人们,他们正在让这美好的事情发生!

版权声明

本译文仅用于学习、研究和交流目的,欢迎非商业转载。转载请注明出处、译者和众成翻译的完整链接。要获取包含以上信息的本文Markdown源文本,请点击这里

关注微信更多学习资料分享!

JavaScript 的新特性:类的 #private 字段的更多相关文章

  1. 前端笔记之ES678&Webpack&Babel(中)对象|字符串|数组的扩展&函数新特性&类

    一.对象的扩展 1.1对象属性名表达式 ES6可以在JSON中使用[]包裹一个key的名字.此时这个key将用表达式作为属性名(被当做变量求值),这个key值必须是字符串. var a = 'name ...

  2. MongoDB 4.2新特性:分布式事务、字段级加密、通配符索引、物化视图

    MongoDB 4.2已经发布,我们来看看它增加了哪些新特性?分布式事务?数据库加密?通配符索引? 在2019年MongoDB World大会上,CTO Eliot Horowitz介绍了MongoD ...

  3. javascript ES6 新特性之 扩展运算符 三个点 ...

    对于 ES6 新特性中的 ... 可以简单的理解为下面一句话就可以了: 对象中的扩展运算符(...)用于取出参数对象中的所有可遍历属性,拷贝到当前对象之中. 作用类似于 Object.assign() ...

  4. 003.ES2015和ES2016新特性--类.md

    JavaScript使用的是基于原型的OO模型,用对象字面量或者函数来实例化对象,用原型链来实现继承. 这样对于数据传统C++.Java的OO范式的开发者来说,会感到比较困惑,于是从ES2015开始逐 ...

  5. JavaScript ECAMScript5 新特性——get/set访问器

    之前对get/set的理解一直有误,觉得get set 是对象属性方法.看了别人的博客也有很多疑问,今天系统的做了很多测试终于弄明白了.(自己通过看书和写demo测试的,如有不对欢迎大家批评指正) g ...

  6. JavaScript ES6 新特性详解

    JavaScript ES6 带来了新的语法和新的强大功能,使您的代码更现代,更易读 const ,  let and var 的区别: const , let 是 ES6 中用于声明变量的新关键字. ...

  7. 用javascript实现完全的类(private、pubulic等)

    js是面向对象的,但是其不像java一样完全的面向对象,但是利用其灵活性,我们可以使用它进行高度的模拟,来看下面的代码: function Student(name){ this.name=name; ...

  8. oracle 12c 新特性之(相同字段上的多重索引、ddl 日志、限制PGA的大小、分页查询)

    1. 相同字段上的多重索引   在Oracle 12c R1之前,一个字段是无法以任何形式拥有多个索引的.或许有人会想知道为什么通常一个字段需要有多重索引,事实上需要多重索引的字段或字段集合是很多的. ...

  9. javascript ES6 新特性之 Promise,ES7 async / await

    es6 一经推出,Promise 就一直被大家所关注.那么,为什么 Promise 会被大家这样关注呢?答案很简单,Promise 优化了回调函数的用法,让原本需要纵向一层一层嵌套的回调函数实现了横向 ...

随机推荐

  1. JavaScript深拷贝与浅拷贝的理解

    个人是这么理解深拷贝和浅拷贝的:就是假设B复制了A,当修改A时,看B是否会发生变化,如果B也跟着变了,说明这是浅拷贝,拿人手短,如果B没变,那就是深拷贝,自食其力. 一起看看我举的浅拷贝栗子: let ...

  2. let和const在es6中的异同点

    let和const这两个都是声明一个变量或函数的方法与var差不太多的效果 let的声明在for循环中,当你定义的是多少,最后你的值就是多少开始的,它只进行一次循环,不会像var那样去一遍一遍的去遍历 ...

  3. (79)zabbix key总是not supported的解决方法

    zabbix定义好key之后,总是会出现Not supported,看到这个问题,大家不用着急,问题其实很容易解决,首先鼠标点击当前key的大红叉上,会显示出报错内容. 常见的有: 1. zabbix ...

  4. 第三章JavaScript 内置对象

    1 Number 1.1 属性 MAX_VALUE JS可以表示的最大的数字 MIN_VALUE JS可以表示的最小的数字 1.2 方法 toFixed(length) 指定保留长度的小数 toExp ...

  5. JavaScript 循环

    for循环:  如果您希望一遍又一遍运行相同的代码,并且每次的值都不同,那么使用循环是很方便的. 我们可以这样输出数组的值: document.write(cars[0] + "<br ...

  6. php扩展开发-哈希表

    什么是哈希表呢?哈希表在数据结构中也叫散列表.是根据键名经过hash函数计算后,映射到表中的一个位置,来直接访问记录,加快了访问速度.在理想情况下,哈希表的操作时间复杂度为O(1).数据项可以在一个与 ...

  7. Python Map, Filter and Reduce

    所属网站分类: python基础 > 函数 作者:慧雅 原文链接: http://www.pythonheidong.com/blog/article/21/ 来源:python黑洞网 www. ...

  8. Django2.2使用mysql数据库pymysql版本不匹配问题的解决过程与总结

    前置条件 django版本:2.2.1 python版本:3.6.6 mysql版本:mysql-community8.0.15 问题 在搭建django项目,配置mysql数据库时遇到无法迁移数据库 ...

  9. (转)iOS静态库与动态库的区别

    一.什么是库? 库是共享程序代码的方式,一般分为静态库和动态库. 静态库:链接时完整地拷贝至可执行文件中,被多次使用就有多份冗余拷贝. 动态库:链接时不复制,程序运行时由系统动态加载到内存,供程序调用 ...

  10. V4L2学习(二)结构介绍

    v4l2_device v4l2_device在v4l2框架中充当所有v4l2_subdev的父设备,管理着注册在其下的子设备.以下是v4l2_device结构体原型(去掉了无关的成员): struc ...