熟悉JavaScript的Map和WeakMap

Map

Map的键/值可以是任何类型

基本API

初始化映射

//使用new关键字和Map构造函数进行初始化
const m1 = new Map(); //使用嵌套数组初始化映射
const m2 = new Map([
['k1', 'v1'],
['k2', 'v2'],
['k3', 'v3']
]); // 使用自定义迭代器初始化
const m3 = new Map({
[Symbol.iterator]:function*() {
yield ['k1', 'v1'];
yield ["k2", "v1"];
yield ["k3", "v3"];
}
})

Map的方法:set()get()has()delete()clear()

Map的属性:size

m2.set('k4', 'v4');
console.log(m2.has('k4'));//true
console.log(m2.get('k4'));//v4
m2.delete('k4');
console.log(m2.has("k4")); //false
console.log(m2.get("k4"));//undefined
m2.clear();
console.log(m2.has('k1'));//false

set()返回映射实例,于是我们可以链式调用set():

m2.set('k4','v4')
.set('k5','v5');

Map的键的相等比较

键的比较基于零值相等(SameValueZero),它认为+0与-0、NaN与NaN是相等的,其他的所有键通过===来比较相等性。

m2.set(NaN, 1);
m2.set(NaN,2);
console.log(m2.get(NaN));//2 m2.set(-0, 3);
console.log(m2.get(+0));//3

Map和Object的比较

Object也可以用来完成映射的作用,那选Map还是Object呢?

从以下方面决策:

  1. 内存占用:Object和Map在不同浏览器上的实现有差异,但是都会随着键值对增加占用内存线性增长,同时在给定相同的内存情况下,Map能比Object多存储50%的键值对。
  2. 插入性能:Map的插入比Object稍微好一点儿。
  3. 查找速度:如果代码涉及大量查找操作,那么选择Object会更好。
  4. 删除性能:如果代码涉及大量删除操作,那么选择Map会更好。
  5. 迭代:Map实现了迭代协议,Object没有。
  6. size:Map的键值对数量可以通过size属性轻松获得,而Object只能手动计算。
  7. 序列化和解析:Map没有序列化和解析支持。

可以实现Map的序列化和解析:How do you JSON.stringify an ES6 Map?

function replacer(key, value) {
if (value instanceof Map) {
return {
dataType: "Map",
value: Array.from(value.entries()), // or with spread: value: [...value]
};
}
return value;
}
function reviver(key, value) {
if (typeof value === "object" && value !== null) {
if (value.dataType === "Map") {
return new Map(value.value);
}
}
return value;
}
const originalValue = new Map([["a", 1]]);
const str = JSON.stringify(originalValue, replacer);
const newValue = JSON.parse(str, reviver);
console.log(originalValue, newValue);

WeakMap

WeakMap称为弱映射。"弱"指的是内存回收机制对待WeakMap中键的方式。

基本API

弱映射的键只能是对象类型(Object或继承自Object)。值类型无限制。

方法和属性的名称和Map相同,除了没有clear()。

弱键

WeakMap中的键不是正式的引用,不会阻止垃圾回收。只要键存在,键值对就会存在于映射中。

const wm = new WeakMap();
wm.set({}, 'v1');

wm新建的键值对的键对象没有被引用,所以垃圾回收机制会回收它。

const wm = new WeakMap();
let obj = new Object();
wm.set(obj, 'v1');
console.log(wm.has(obj));//true
obj = null;
console.log(wm.has(obj));//false

不可迭代键

WeakMap没有提供迭代键值对的功能,也没有必要,是为了保证只有通过键对象的引用才能取得值。

弱映射的实际使用

私有变量

以对象实例为键,以私有成员的字典为值。

const wm = new WeakMap();
class User{
constructor(id) {
this.userId = Symbol('id');
this.setId(id);
}
setId(id) {
const privateMembers = wm.get(this) || {};
privateMembers[this.userId] = id;
wm.set(this, privateMembers);
}
getId() {
return wm.get(this)[this.userId];
}
}
const user = new User(1);
console.log(user.getId());//1
user.setId(2);
console.log(user.getId());//2

上面的变量userId并非真正私有,通过弱映射vm可以获取:

console.log(wm.get(user)[user.userId]);//2

我们可以把弱映射和类放在一个闭包里:

const User = (() => {
const wm = new WeakMap();
class User {
constructor(id) {
this.userId = Symbol("id");
this.setId(id);
}
setId(id) {
const privateMembers = wm.get(this) || {};
privateMembers[this.userId] = id;
wm.set(this, privateMembers);
}
getId() {
return wm.get(this)[this.userId];
}
}
return User;
})();
const user = new User(1);
console.log(user.getId());//1
user.setId(2);
console.log(user.getId());//2

JavaScript的Map和WeakMap的更多相关文章

  1. javascript之Map

    javascript中的map,我用的不是特别多,倒是Java中的Map或HashMap,经常用. 顺便围绕几个方面介绍一下map? 1.Map对象 Map对象是一种有对应键值对的对象,JS的Obje ...

  2. JavaScript之Map对象

    前言 工欲善其事,必先利其器.这是一款以前在前端项目中没有使用过的.有趣的对象,咱来看看如何使用~ 并非arrayObj.map(function) //arrayObj.map与arrayObj.f ...

  3. Map 和 WeakMap 数据结构

    Map 和 WeakMap 是ES6 新增的数据结构 一.Map 它们本质与对象一样,都是键值对的集合,但是他们与 Object 对象主要的不同是,键可以是各种类型的数值,而Object 对象的键 只 ...

  4. JavaScript之map与parseInt的陷阱

    问题来源 ​ 这个问题的来源是学习廖雪峰老师JS教程.问题如下:小明希望利用map()把字符串变成整数,他写的代码很简洁: 'use strict'; var arr = ['1', '2', '3' ...

  5. Map和WeakMap的区别

    个人总结:在一个变量作用域中,如果结束到作用域结尾 } 的话,map中的引用会被垃圾回收机制回收的是weakmap ,map中的引用不会被垃圾回收机制回收的是map. 强引用:只要引用存在,垃圾回收器 ...

  6. javascript自定义Map对象

    javascript定义map对象开发前端组件的重要性就不过多阐述了,直接参考以下案例即可 <script type=text/javascript charset=utf-8> func ...

  7. JavaScript的Map和Set

    JavaScript的Map和Set 1.map:映射(通过key获得value).增.删 2.set:增.删.判断是否包含某个元素 1. Map <!DOCTYPE html><h ...

  8. ES6新增的Map和WeakMap 又是什么玩意?非常详细的解释

    上一篇文章讲了set和weakSet,这节咱就讲Map和weakMap是什么?这两篇文章并没有什么联系,主要知识用法类似而已.嘿嘿,是不是感觉舒服多了. 什么是Map 介绍什么是Map,就不得不说起O ...

  9. Map和WeakMap的方法和区别

    Map Map是一组键值对的结构,具有极快的查找速度. 一.构造函数不同 let map = new Map(); let weakmap = new WeakMap(); 二.内置函数不同 Map的 ...

  10. ES6新特性:Javascript中的Map和WeakMap对象

    Map对象 Map对象是一种有对应 键/值 对的对象, JS的Object也是 键/值 对的对象 : ES6中Map相对于Object对象有几个区别: 1:Object对象有原型, 也就是说他有默认的 ...

随机推荐

  1. 2022-11-06:给定平面上n个点,x和y坐标都是整数, 找出其中的一对点的距离,使得在这n个点的所有点对中,该距离为所有点对中最小的。 返回最短距离,精确到小数点后面4位。

    2022-11-06:给定平面上n个点,x和y坐标都是整数, 找出其中的一对点的距离,使得在这n个点的所有点对中,该距离为所有点对中最小的. 返回最短距离,精确到小数点后面4位. 答案2022-11- ...

  2. 2021-08-19:超级洗衣机。假设有 n 台超级洗衣机放在同一排上。开始的时候,每台洗衣机内可能有一定量的衣服,也可能是空的。在每一步操作中,你可以选择任意 m (1 ≤ m ≤ n) 台洗衣机,

    2021-08-19:超级洗衣机.假设有 n 台超级洗衣机放在同一排上.开始的时候,每台洗衣机内可能有一定量的衣服,也可能是空的.在每一步操作中,你可以选择任意 m (1 ≤ m ≤ n) 台洗衣机, ...

  3. web自动化04-css定位

    css元素定位 1. 是什么? 用来描述html元素的显示样式 选择器是一种模式,用于选择需要添加样式的元素   selenium中推荐使用css定位,比XPath定位要快    2.如何定位?   ...

  4. ggplot2 调整绘图区域大小

    熟悉 R 绘图的朋友肯定知道,在普通绘图中,图片的大小可以直接在 png() 和 pdf() 中指定,而绘图区大小则可以用 par() 中的 mar 或 mai 来指定.但是在 ggplot2 中,图 ...

  5. UpSetR 关联的 venneuler 包安装笔记

    本文章已经设置了最低额度的付费阅读,如果您觉得文章对您有用,且手头宽裕,欢迎请作者喝杯热茶.本文章付费部分内容并不影响您对文章的阅读和理解,只是作者对付费阅读的一次尝试,感谢. 背景 R 语言中的 v ...

  6. SQL Server 日志传输还原作业执行缓慢

    目录 情景 故障定位 VLF 对 Restore 的影响 问题 解决方案 方案1 方案 2 参考资料 情景 IP 角色 192.168.1.61 Primary 192.168.1.59 Second ...

  7. 一文了解Go语言的I/O接口设计

    1. 引言 I/O 操作在编程中扮演着至关重要的角色.它涉及程序与外部世界之间的数据交换,允许程序从外部,如键盘.文件.网络等地方读取数据,也能够将外界输入的数据重新写入到目标位置中.使得程序能够与外 ...

  8. Auto.js食用指南

    Auto.js食用指南 控件点击是autojs特有的一项功能,基于安卓的无障碍功能的,在软件上有很好的支持,常用于办公软件等...... 前言: 软件选择: auto.js 8.0pro版本(对比4. ...

  9. msyql的基本操作(增删改查)

    一.SQL 语法要点 1.SQL 语句不区分大小写,但是数据库表名.列名和值是否区分,依赖于具体的 DBMS 以及配置. 例如:SELECT 与 select .Select 是相同的. 2.多条 S ...

  10. 利用python分析pdf数据,分析上市公司财报

    import re import os.path import matplotlib import matplotlib.pyplot as plt from pdfminer.pdfparser i ...