当前位置: 首页 > news >正文

JS【详解】Map (含Map 和 Object 的区别,Map 的常用 API,Map与Object 的性能对比,Map 的应用场景和不适合的使用场景)

Map 简介

ES6 新增了数据结构 Map,与普通对象 object 类似,也是键值对的集合,特征如下:

  • 有序(将元素添加的先后顺序作为其次序)【object 无序】
  • 键可以是任意数据类型 【object 的键只能是字符串/Symbol】
  • Map是一个纯哈希结构,而Object不是(它拥有自己的内部逻辑)。
  • 键是唯一性的,不存在重复的键值对。(与 object 相同)
  • 因 Map 也是对象类型,所以它也是 Object 实例
// person_m 是一个Map对象
typeof person_m  // 得到 object
console.log(person_m instanceof Object); //true

Map 和 Object 的区别

  • Map 有序,Object 无序
  • Map 的键可以是任意数据类型,Object 的键只能是字符串/Symbol
  • Map是一个纯哈希结构,而Object不是(它拥有自己的内部逻辑)。
  • Map 只能通过构造函数的方式创建,Object 可以通过多种方式(字面量、new Object()、Object.create() 等)创建
  • Map本身具有size属性,Object需要使用 keys()、values()等方法获取;
  • Map本身具有可迭代属性,Object不具有;
  • Map会保持数据的插入顺序,Object不会;

Map 的基本操作

new —— 创建 Map

创建空Map对象

let blank_m = new Map();
console.log(blank_m);
// 打印 Map(0) {}

创建含键值对的 Map 对象

let person_m = new Map([["name", "朝阳"],["age", 35],
]);
console.log(person_m);
//  打印 Map(2) { 'name' => '朝阳', 'age' => 35 }

创建以对象为键的 Map 对象

let cp1_boy = {name: "朝阳",
};let cp1_girl = {name: "晚霞",
};let cp2_boy = {name: "范闲",
};let cp2_girl = {name: "林婉儿",
};let cps = new Map([[cp1_boy, cp1_girl],[cp2_boy, cp2_girl],
]);
console.log(cps);
// 打印
// Map(2) {
//   { name: '朝阳' } => { name: '晚霞' },
//   { name: '范闲' } => { name: '林婉儿' }
// }

get —— 读取键的值

person_m.get("name")

set —— 新增键值对 / 修改键的值

若不存在目标键则新增,若存在,则修改目标键的值

person_m.set("gender", "男");

支持链式调用

person_m.set("age", 40).set("gender", "男");

delete —— 删除键值对

person_m.delete("name");

object 删除键的本质是将其键的值设置为 undefined,所以其键其实仍在内存中!

delete target_obj.name

clear —— 清空键值对

person_m.clear()

has —— 是否存在目标键

返回 true / false

if (person_m.has("name")) {
}

size —— 获取键值对的数量

类似数组的 .length

person_m.size

object 的方法为

Object.keys(target_obj).length

Object.getOwnPropertyNames(target_obj).length

遍历 Map

forEach ———— 留意顺序 value, key

person_m.forEach((value, key) => {console.log(key, value);
})

for of ———— 留意顺序 key,value (使用了数组解构赋值 [ ] !)

for(let [key,value] of person_m){console.log(key,value);
}

Map 转 数组

let arr=[...person_m]

得到

[ [ 'name', '朝阳' ], [ 'age', 35 ] ]

其他操作

  • 获取所有键 keys()
for(let key of person_m.keys()){console.log(key);
}
  • 获取所有值 values()
for(let value of person_m.values()){console.log(value);
}
  • 获取所有键和属性值 entries()
for(let [key,value] of person_m.entries()){console.log(key,value);
}

Map 的性能(与 Object 对比)

  • 增加键
    量小时差异不大,量大时,Map比object 快,因为 Map内部使用了哈希表,而object在内存分配和处理上开销较大

  • 查找键
    都非常快,但在某些情况下,当查找的键在数据结构中不存在时,Map可能会比 Object 更快。这是因为Map会在内部快速定位到哈希表中的空位,而0bject需要遍历整个原型链。

  • 删除键
    Map 优于 Object 。因为Map的内部实现使得删除操作更加高效。在Object中,删除一个键实际上仅仅是将其值设置为undefined,而不是真正地从内存中移除,这可能导致内存泄漏和性能下降。

  • 遍历
    Map 优于 Object 。Map内部的哈希表结构保证了遍历的顺序与插入顺序相同。而Object在遍历时,可能因为属性的插入顺序和内部哈希结构而导致遍历顺序不一致。此外,Map提供了内置的迭代器,可以使用for…of循环方便地遍历键值对。

Map 的应用场景

关联复杂数据类型的数据

利用 Map 的键可为任意数据类型的特性!

const employeeProjects = new Map();
employeeProjects.set(employee1, [projectA, projectB]);
employeeProjects.set(employee2, [projectC, projectD]);// 查找员工参与的项目
console.log(employeeProjects.get(employee1)); // 输出 [projectA, projectB]

Vue3 数据响应式 reactive 和 effect 的源码中,大量使用了 Map

大量键的增删和查找操作

在键的量较大时,Map 对键的增加、删除、查找和遍历性能都比 object 更好。

Map 不适合使用的场景

  • 在需要使用JSON时(因JSON 尚未支持Map,但直接支持Object)

相关文章:

  • mysql 删除重复数据 关联自己 关联子查询 delete
  • 掌握ASPICE标准:汽车软件测试工程师的专业发展路径
  • vue 笔记02
  • C++ | Leetcode C++题解之第117题填充每个节点的下一个右侧节点指针II
  • 大模型中GPTs,Assistants API, 原生API的使用场景?
  • 数据分析中的列与行交换技巧
  • 【Android14 ShellTransitions】(一)开篇
  • 【乐吾乐3D可视化组态编辑器】模型类型与属性
  • IP 分片过程及偏移量计算
  • 多模态大模型:系统、趋势与问题
  • 对于个人而言,大数据时代如何更好地管理自己的信息?
  • Python中使用“import”还是“from ... import”导入模块
  • 双机多网口配置同网段地址,可以通过目的IP确定接收数据的网卡吗?
  • C#算法(15)—求四边形的外接矩形
  • 部署Prometheus + Grafana实现监控数据指标
  • -------------------- 第二讲-------- 第一节------在此给出链表的基本操作
  • 《深入 React 技术栈》
  • 「前端」从UglifyJSPlugin强制开启css压缩探究webpack插件运行机制
  • 【vuex入门系列02】mutation接收单个参数和多个参数
  • 【腾讯Bugly干货分享】从0到1打造直播 App
  • 10个确保微服务与容器安全的最佳实践
  • android图片蒙层
  • CSS中外联样式表代表的含义
  • Docker 1.12实践:Docker Service、Stack与分布式应用捆绑包
  • E-HPC支持多队列管理和自动伸缩
  • FastReport在线报表设计器工作原理
  • JavaScript类型识别
  • Mybatis初体验
  • mysql_config not found
  • Netty 4.1 源代码学习:线程模型
  • Odoo domain写法及运用
  • React的组件模式
  • spring boot下thymeleaf全局静态变量配置
  • springboot_database项目介绍
  • spring学习第二天
  • 初探 Vue 生命周期和钩子函数
  • 多线程事务回滚
  • 微信小程序上拉加载:onReachBottom详解+设置触发距离
  • 小白应该如何快速入门阿里云服务器,新手使用ECS的方法 ...
  • ​LeetCode解法汇总518. 零钱兑换 II
  • ​你们这样子,耽误我的工作进度怎么办?
  • # .NET Framework中使用命名管道进行进程间通信
  • # Redis 入门到精通(一)数据类型(4)
  • #Datawhale AI夏令营第4期#AIGC文生图方向复盘
  • ${factoryList }后面有空格不影响
  • $emit传递多个参数_PPC和MIPS指令集下二进制代码中函数参数个数的识别方法
  • ( )的作用是将计算机中的信息传送给用户,计算机应用基础 吉大15春学期《计算机应用基础》在线作业二及答案...
  • (1)(1.11) SiK Radio v2(一)
  • (13)[Xamarin.Android] 不同分辨率下的图片使用概论
  • (2)从源码角度聊聊Jetpack Navigator的工作流程
  • (4) openssl rsa/pkey(查看私钥、从私钥中提取公钥、查看公钥)
  • (8)Linux使用C语言读取proc/stat等cpu使用数据
  • (C语言)逆序输出字符串
  • (pytorch进阶之路)CLIP模型 实现图像多模态检索任务
  • (分享)自己整理的一些简单awk实用语句