[JavaScript] private field & weak map

1. 背景

private fieldstc39 stage 3的提案,

class T {
  #x = 0;

  addOne(){
    return ++this.#x;
  }
}

const t = new T;
Reflect.ownKeys(t);  // []

private fields以#号开头,例如,以上代码中,#x就是一个private fields,
我们使用Reflect.ownKeys获取实例t的所有属性名,结果是一个空列表。

注:

v = {};

// 可枚举属性
v['a'] = 1;

// 不可枚举属性
Object.defineProperty(v, 'b', {
  value: 2,
  enumerable: false,
});

// Symbol属性
s = Symbol('c');
v[s] = 3;

Object.keys(v);  // ['a']
Object.getOwnPropertyNames(v);  // ['a', 'b']
Object.getOwnPropertySymbols(v);  // [Symbol(c)]
Reflect.ownKeys(v);  // ["a", "b", Symbol(c)]

(1)Object.keys会获取对象自身的可枚举属性
(2)Object.getOwnPropertyNames会获取对象自身的可枚举属性不可枚举属性
(3)Reflect.ownKeys会获取对象自身的可枚举属性不可枚举属性Symbol属性

2. 用babel进行编译

babelv7.0.0-beta.48之后,增加了对private fields的支持,
我们来编译上面的代码,查看一下编译结果。

(1)全局安装@babel/cli,和@babel/core,并指定版本

$ npm i -g @babel/cli@7.0.0-beta.48 @babel/core@7.0.0-beta.48
$ babel --version
> 7.0.0-beta.48 (@babel/core 7.0.0-beta.48)

(2)新建npm项目,并初始化

$ mkdir test-private-field
$ cd test-private-field
$ npm init

(3)新建.babelrc文件

{
  "presets": [
    "@babel/preset-env",
    "@babel/preset-stage-3"
  ]
}

(4)安装开发环境依赖

$ npm i -D \
@babel/core@7.0.0-beta.48 \
@babel/preset-env@7.0.0-beta.48 \
@babel/preset-stage-3@7.0.0-beta.48

(5)源码index.js

class T {
  #x = 0;

  addOne(){
    return ++this.#x;
  }
}

const t = new T;
Reflect.ownKeys(t);  // []

(6)使用babel编译,输出到dist.js

$ babel index.js > dist.js

(7)编译结果如下

"use strict";

function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }

function _defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } }

function _createClass(Constructor, protoProps, staticProps) { if (protoProps) _defineProperties(Constructor.prototype, protoProps); if (staticProps) _defineProperties(Constructor, staticProps); return Constructor; }

function _classPrivateFieldSet(receiver, privateMap, value) { if (!privateMap.has(receiver)) { throw new TypeError("attempted to set private field on non-instance"); } privateMap.set(receiver, value); return value; }

function _classPrivateFieldGet(receiver, privateMap) { if (!privateMap.has(receiver)) { throw new TypeError("attempted to get private field on non-instance"); } return privateMap.get(receiver); }

var T =
/*#__PURE__*/
function () {
  function T() {
    _classCallCheck(this, T);

    _x.set(this, 0);
  }

  _createClass(T, [{
    key: "addOne",
    value: function addOne() {
      return _classPrivateFieldSet(this, _x, +_classPrivateFieldGet(this, _x) + 1);
    }
  }]);

  return T;
}();

var _x = new WeakMap();

var t = new T();
Reflect.ownKeys(t); // []

我们看到private fields是用WeakMap实现的。

3. WeakMap

WeakMap是ECMAScript 2015的特性,它与Map是不同的。
(1)WeakMap的键必须是一个对象(object),
(2)并且WeakMap对其自身的键是弱引用的(weakly referenced)。

所谓弱引用,指的是作为WeakMap键的对象,一旦没有其他对象引用它,
该键值对就会失效,被当做垃圾回收。

由于WeakMap的这个特性,我们就可以用它来实现private fields了。

const weakMap = new WeakMap;
class T {
  constructor(){
    weakMap.set(this, {
      x: 0,
    });
  }

  addOne(){
    const fields = weakMap.get(this);
    return ++fields.x;
  }
}

const t = new T;
Reflect.ownKeys(t);  // []

以上代码中,每次创建一个T的实例,都会向weakMap增加一个键值对,
其中,键为当前创建的T的实例t,值为这个实例所拥有的private fields。

T的实例t不被其他对象引用的时候,该键值对会被自动回收,
而如果使用Map的话,则必须手动回收。


参考

MDN: Reflect.ownKeys
MDN: Object.getOwnPropertySymbols
babel: v7.0.0-beta.48
MDN: WeakMap

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 136,174评论 19 139
  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 5,890评论 0 9
  • 前言 人生苦多,快来 Kotlin ,快速学习Kotlin! 什么是Kotlin? Kotlin 是种静态类型编程...
    任半生嚣狂阅读 26,549评论 9 118
  • 对象的创建与销毁 Item 1: 使用static工厂方法,而不是构造函数创建对象:仅仅是创建对象的方法,并非Fa...
    孙小磊阅读 6,338评论 0 3
  • 我很冷,一个很不恰当的词,很不符合季节的感受,脚底湿冷湿冷,仿佛回到了冬天。 我一直觉得自己的身体很有智行,...
    眼神犀利的猫阅读 1,747评论 0 0

友情链接更多精彩内容