WeakMap 和 WeakSet:解决内存泄漏避免循环引用(上)

2023-12-22 06:28

本文主要是介绍WeakMap 和 WeakSet:解决内存泄漏避免循环引用(上),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在这里插入图片描述

🤍 前端开发工程师(主业)、技术博主(副业)、已过CET6
🍨 阿珊和她的猫_CSDN个人主页
🕠 牛客高级专题作者、在牛客打造高质量专栏《前端面试必备》
🍚 蓝桥云课签约作者、已在蓝桥云课上架的前后端实战课程《Vue.js 和 Egg.js 开发企业级健康管理项目》、《带你从入门到实战全面掌握 uni-app》

文章目录

  • 一、引言
    • 介绍WeakMap 和 WeakSet 的背景和用途
  • 二、 WeakMap 简介
    • 定义和语法
    • 与普通 Map 的区别
    • WeakMap 的键必须是对象,值可以是任意类型
    • 如何创建和使用 WeakMap
  • 三、 WeakSet 简介
    • 定义和语法
    • 与普通 Set 的区别
    • WeakSet 只包含对象
    • 如何创建和使用 WeakSet

一、引言

介绍WeakMap 和 WeakSet 的背景和用途

以下是WeakMap和WeakSet的背景和用途介绍:

WeakMap类似于Map,但是键只能是对象类型且键名所指向的对象是弱引用,这意味着如果这个对象在其他地方没有被引用,那么它将会被垃圾回收,这也是WeakMap的主要应用场景。

WeakSet类似于Set,但成员只能是对象类型,且成员对象是弱引用,这意味着如果这个对象在其他地方没有被引用,那么它将会被垃圾回收,这也是WeakSet的主要应用场景。

总的来说,WeakMap和WeakSet主要用于解决内存泄漏问题,避免循环引用,以及在一些临时数据存储和缓存的场景中提高性能。

二、 WeakMap 简介

定义和语法

WeakMap是ES6中新增的一种数据结构,用于存储键值对。它与Map类似,但有一些重要区别:

  • 键必须是对象类型,值可以是任意类型。
  • 由于键是弱引用,如果对键的所有引用都丢失,并且不再有对值的引用,则该值可能会被垃圾回收。

以下是WeakMap的定义和语法示例:

new WeakMap()  // 创建一个新的WeakMap对象
new WeakMap(iterable)  // iterable是数组或者任意可以迭代的对象,需要拥有key-value对(一般是一个二维数组)

WeakMap 实例方法包括:

  • delete(key):删除与键关联的任意值。删除后,has(key)返回false
  • get(key):返回与键关联的值,如果不存在关联值,则返回undefined
  • has(key):返回键在 WeakMap 上是否存在的结果。
  • set(key, value):在 WeakMap 对象上为对应键设置指定的值,并返回 WeakMap 对象。

与普通 Map 的区别

WeakMap 与普通 Map 的主要区别在于它们对键的处理方式不同。

普通 Map 的键必须是可枚举的(即能够通过for...of循环遍历),并且是强引用的。这意味着只要有对 Map 中键的引用存在,这个键就不会被垃圾回收。

而 WeakMap 的键必须是对象类型,并且是弱引用的。这意味着如果除了WeakMap 本身之外,没有其他对键的引用存在,那么这个键和对应的值都可能会被垃圾回收。

以下是一个示例,展示了 WeakMap 与普通 Map 在键的生命周期上的区别:

// 创建一个普通 Map
const map = new Map();
const obj1 = {key: 'value1'};
map.set(obj1, 'data');// 创建一个 WeakMap
const weakMap = new WeakMap();
const obj2 = {key: 'value2'};
weakMap.set(obj2, 'data');// 断开对 obj1 和 obj2 的引用
obj1 = null;
obj2 = null;// 检查普通 Map 中键是否存在
console.log(map.has(obj1)); 
console.log(map.get(obj1)); // 检查 WeakMap 中键是否存在
console.log(weakMap.has(obj2)); 
console.log(weakMap.get(obj2)); 

在这个示例中,我们创建了一个普通 Map 和一个 WeakMap,并将两个对象作为键分别存储在它们里面。然后,我们断开了对这两个对象的引用。

对于普通 Map,由于键是强引用的,即使我们断开了对对象的引用,键仍然存在于 Map 中,并且可以通过map.has(obj1)map.get(obj1)来检查和获取对应的值。

对于 WeakMap,由于键是弱引用的,当我们断开了对对象的引用后,WeakMap 无法检测到键的存在,因此weakMap.has(obj2)返回false,并且weakMap.get(obj2)返回undefined

这就是 WeakMap 和普通 Map 在键的处理方式上的主要区别。WeakMap 适用于在某些情况下需要避免内存泄漏的场景,例如缓存、临时数据存储等。

WeakMap 的键必须是对象,值可以是任意类型

以下是一个使用 WeakMap 的简单代码示例:

// 创建一个 WeakMap 对象
const weakMap = new WeakMap();// 创建两个对象作为 WeakMap 的键
const obj1 = { key: 1 };
const obj2 = { key: 2 };// 将值与对应的对象键关联起来
weakMap.set(obj1, 'value1');
weakMap.set(obj2, 'value2');// 检查键是否存在
console.log(weakMap.has(obj1));  
console.log(weakMap.has(obj2)); // 获取对应键的值
console.log(weakMap.get(obj1));  
console.log(weakMap.get(obj2));  // 删除键值对
weakMap.delete(obj1);// 检查键是否存在(现在应该为 false)
console.log(weakMap.has(obj1));  

在这个示例中,我们创建了一个 WeakMap 对象,并将两个对象作为键与一些值关联起来。然后,我们检查键是否存在,并获取对应的值。最后,我们删除了一个键值对,并再次检查该键是否存在。

请注意,WeakMap 的键必须是对象,而值可以是任意类型。另外,WeakMap 不会阻止垃圾回收对键的回收,因此如果没有其他对键的引用,对应的键值对可能会被自动删除。

如何创建和使用 WeakMap

WeakMap 是一种用于从外部扩展对象而不干扰垃圾回收的 JavaScript 数据结构。它是一个 Map 字典,其中的键很弱,即如果对该键的所有引用都丢失,并且不再有对该值的引用,则可以对该值进行垃圾回收。

与普通 Map 相比,WeakMap 的键必须是对象,而值可以是任意类型。

创建和使用 WeakMap 的方法如下:

let weakMap = new WeakMap();
let o = {n: 1};
weakMap.set(o, "A"); // 添加
console.log(weakMap.has(o)); // 检查键是否存在
console.log(weakMap.get(o)); // 检索与键关联的值

三、 WeakSet 简介

定义和语法

WeakSet 是ES6中引入的一种新的数据结构,它类似于数组,但是成员的值都是唯一的,没有重复的值。与Set不同的是,WeakSet 的成员都是对象且都是弱引用,这意味着它们可以被垃圾回收机制回收,因此可以用来保存 DOM 节点,不容易造成内存泄漏

WeakSet 的语法与Set类似,也是一个构造函数,可以使用new命令创建WeakSet数据结构。作为构造函数,WeakSet可以接受一个数组或类似数组的对象作为参数。

WeakSet 结构有以下三个方法:

  • WeakSet.prototype.add(value): 向WeakSet实例添加一个新成员。
  • weakSet.prototype.delete(value): 清除WeakSet实例的指定成员。
  • weakSet.prototype.has(value): 返回一个布尔值,表示某个值是否在WeakSet实例中。

需要注意的是,WeakSet 不能遍历,因为成员都是弱引用,随时可能会消失,遍历机制无法保证成员存在。

与普通 Set 的区别

WeakSet 与普通 Set 的主要区别在于它们对成员的存储方式不同。

普通 Set 的成员是强引用的,这意味着只要 Set 中存在对某个对象的引用,这个对象就不会被垃圾回收器回收。

而 WeakSet 的成员是弱引用的,这意味着如果除了 WeakSet 之外没有其他对某个对象的引用,那么这个对象可能会被垃圾回收器回收。

以下是一个示例,展示了 WeakSet 与普通 Set 在成员的生命周期上的区别:

// 创建一个普通 Set
const strongSet = new Set();
const obj1 = {key: 'value1'};
strongSet.add(obj1);// 创建一个 WeakSet
const weakSet = new WeakSet();
const obj2 = {key: 'value2'};
weakSet.add(obj2);// 断开对 obj1 和 obj2 的引用
obj1 = null;
obj2 = null;// 检查普通 Set 中键是否存在
console.log(strongSet.has(obj1)); 
console.log(strongSet.has(obj2)); // 检查 WeakSet 中键是否存在
console.log(weakSet.has(obj1)); 
console.log(weakSet.has(obj2)); 

在这个示例中,我们创建了一个普通 Set 和一个 WeakSet,并将两个对象作为成员添加到它们里面。然后,我们断开了对这两个对象的引用。

对于普通 Set,由于成员是强引用的,即使我们断开了对对象的引用,Set 仍然能够检测到成员的存在,因此strongSet.has(obj1)strongSet.has(obj2)仍然返回true

对于 WeakSet,由于成员是弱引用的,当我们断开了对对象的引用后,WeakSet 无法检测到成员的存在,因此weakSet.has(obj1)weakSet.has(obj2)返回false

这就是 WeakSet 和普通 Set 在成员的生命周期上的主要区别。WeakSet 适用于在某些情况下需要避免内存泄漏的场景,例如缓存、临时数据存储等。

WeakSet 只包含对象

WeakSet 只包含对象,而不能包含其他类型的值,例如原始类型(如数字、字符串等)或函数等。如果试图向 WeakSet 添加非对象类型的值,将会抛出一个 TypeError 错误。

这是因为 WeakSet 中的对象都是弱引用,即垃圾回收机制不考虑 WeakSet 对该对象的引用,也就是说,如果其他对象都不再引用该对象,那么垃圾回收机制会自动回收该对象所占用的内存,而不考虑该对象还存在于 WeakSet 之中。

如何创建和使用 WeakSet

要创建一个WeakSet,可以使用new关键字和WeakSet构造函数,示例如下:

const weakSet = new WeakSet();

由于WeakSet的元素必须是对象,而不能是原始类型(如字符串、数字等),并且WeakSet不可迭代,因此不能使用for...of循环遍历其中的元素,也没有size属性和forEach方法。

WeakSet的主要方法有add()delete()has(),用于向WeakSet实例添加新成员、清除指定成员以及判断某个值是否在WeakSet实例中。

由于WeakSet的成员都是弱引用,随时可能消失,因此遍历机制无法保证成员存在。WeakSet适合用于临时存放一组对象以及存放跟对象绑定的信息,只要这些对象在外部消失,它在WeakSet里面引用的就会自动消失,可以更好地管理对象的生命周期并避免内存泄漏的问题。

这篇关于WeakMap 和 WeakSet:解决内存泄漏避免循环引用(上)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/522879

相关文章

qt5cored.dll报错怎么解决? 电脑qt5cored.dll文件丢失修复技巧

《qt5cored.dll报错怎么解决?电脑qt5cored.dll文件丢失修复技巧》在进行软件安装或运行程序时,有时会遇到由于找不到qt5core.dll,无法继续执行代码,这个问题可能是由于该文... 遇到qt5cored.dll文件错误时,可能会导致基于 Qt 开发的应用程序无法正常运行或启动。这种错

C++高效内存池实现减少动态分配开销的解决方案

《C++高效内存池实现减少动态分配开销的解决方案》C++动态内存分配存在系统调用开销、碎片化和锁竞争等性能问题,内存池通过预分配、分块管理和缓存复用解决这些问题,下面就来了解一下... 目录一、C++内存分配的性能挑战二、内存池技术的核心原理三、主流内存池实现:TCMalloc与Jemalloc1. TCM

SpringBoot排查和解决JSON解析错误(400 Bad Request)的方法

《SpringBoot排查和解决JSON解析错误(400BadRequest)的方法》在开发SpringBootRESTfulAPI时,客户端与服务端的数据交互通常使用JSON格式,然而,JSON... 目录问题背景1. 问题描述2. 错误分析解决方案1. 手动重新输入jsON2. 使用工具清理JSON3.

MySQL存储过程之循环遍历查询的结果集详解

《MySQL存储过程之循环遍历查询的结果集详解》:本文主要介绍MySQL存储过程之循环遍历查询的结果集,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录前言1. 表结构2. 存储过程3. 关于存储过程的SQL补充总结前言近来碰到这样一个问题:在生产上导入的数据发现

MySQL 设置AUTO_INCREMENT 无效的问题解决

《MySQL设置AUTO_INCREMENT无效的问题解决》本文主要介绍了MySQL设置AUTO_INCREMENT无效的问题解决,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参... 目录快速设置mysql的auto_increment参数一、修改 AUTO_INCREMENT 的值。

关于跨域无效的问题及解决(java后端方案)

《关于跨域无效的问题及解决(java后端方案)》:本文主要介绍关于跨域无效的问题及解决(java后端方案),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录通用后端跨域方法1、@CrossOrigin 注解2、springboot2.0 实现WebMvcConfig

Redis过期删除机制与内存淘汰策略的解析指南

《Redis过期删除机制与内存淘汰策略的解析指南》在使用Redis构建缓存系统时,很多开发者只设置了EXPIRE但却忽略了背后Redis的过期删除机制与内存淘汰策略,下面小编就来和大家详细介绍一下... 目录1、简述2、Redis http://www.chinasem.cn的过期删除策略(Key Expir

Go语言中泄漏缓冲区的问题解决

《Go语言中泄漏缓冲区的问题解决》缓冲区是一种常见的数据结构,常被用于在不同的并发单元之间传递数据,然而,若缓冲区使用不当,就可能引发泄漏缓冲区问题,本文就来介绍一下问题的解决,感兴趣的可以了解一下... 目录引言泄漏缓冲区的基本概念代码示例:泄漏缓冲区的产生项目场景:Web 服务器中的请求缓冲场景描述代码

解决JSONField、JsonProperty不生效的问题

《解决JSONField、JsonProperty不生效的问题》:本文主要介绍解决JSONField、JsonProperty不生效的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑... 目录jsONField、JsonProperty不生效javascript问题排查总结JSONField

github打不开的问题分析及解决

《github打不开的问题分析及解决》:本文主要介绍github打不开的问题分析及解决,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、找到github.com域名解析的ip地址二、找到github.global.ssl.fastly.net网址解析的ip地址三