1>意外的全局变量
函数中意外的定义了全局变量,每次执行该函数都会生成该变量,且不会随着函数执行结束而释放。

2>未清除的定时器
定时器没有清除,它内部引用的变量,不会被释放。

3>脱离DOM的元素引用
一个dom容器删除之后,变量未置为null,则其内部的dom元素则不会释放。

4>持续绑定的事件
函数中addEventListener绑定事件,函数多次执行,绑定便会产生多次,产生内存泄漏。

5>绑在EventBus的事件没有解绑

6>闭包引起内存泄漏
比如事件处理回调,导致DOM对象和脚本中对象双向引用。

7>使用第三方库创建,没有调用正确的销毁函数

8>单页应用时,页面路由切换后,内存未释放

JavaScript 有完善的内存处理机制,能自动进行垃圾回收,但是假如一个对象一直被引用,他的内存是无法得到释放的。如果项目运行过程中,内存占用越来越高,只增不减,没有峰值,就存在内存泄漏。多页应用我们可以通过页面刷新缓解,但是对于服务端渲染和单页应用则需要重点关注内存泄漏问题。本文主要以Vue单页应用展开,因为在 SPA 的设计中,用户使用它时是不需要刷新浏览器的,所以 JavaScript 应用需要自行清理组件来确保垃圾回收以预期的方式生效。

一、什么是内存泄漏
程序的运行需要内存。只要程序提出要求,操作系统或者运行时就必须供给内存。对于持续运行的服务进程,必须及时释放不再用到的内存。否则,内存占用越来越高,轻则影响系统性能,重则导致进程崩溃。不再用到的内存,没有及时释放,就叫做内存泄漏。

二、如何判断内存泄漏
Chrome 浏览器memory有3种方式,可以分析内存:

Heap snapshot - 用以打印堆快照,堆快照文件显示页面的 javascript 对象和相关 DOM 节点之间的内存分配

Allocation instrumentation on timeline - 在时间轴上记录内存信息,随着时间变化记录内存信息。

Allocation sampling - 内存信息采样,使用采样的方法记录内存分配。此配置文件类型具有最小的性能开销,可用于长时间运行的操作。它提供了由 javascript 执行堆栈细分的良好近似值分配。

三、如何定位内存泄漏
点击快照-并hover右测的列表项,会显示部分存在内存泄漏的地方

四、常见的内存泄漏
意外的全局变量
函数中意外的定义了全局变量,每次执行该函数都会生成该变量,且不会随着函数执行结束而释放。

未清除的定时器
定时器没有清除,它内部引用的变量,不会被释放。

脱离DOM的元素引用
一个dom容器删除之后,变量未置为null,则其内部的dom元素则不会释放。

持续绑定的事件
函数中addEventListener绑定事件,函数多次执行,绑定便会产生多次,产生内存泄漏。

绑在EventBus的事件没有解绑

闭包引起内存泄漏
比如事件处理回调,导致DOM对象和脚本中对象双向引用。

使用第三方库创建,没有调用正确的销毁函数

单页应用时,页面路由切换后,内存未释放

五、解决内存泄漏
变量先申明后使用。

setTimeout setInterval清理 (最好不用)可以使用nextTick代替。

如果在mounted/created 钩子中绑定了DOM/BOM 对象中的事件,需要在beforeDestroy 中做对应解绑处理。

 1 mounted () {
2 window.addEventListener('resize', this.onResize)
3 },
4 beforeDestroy () {
5 window.removeEventListener('resize', this.onResize)
6 }
7 如果在mounted/created 钩子中使用了on,需要在beforeDestroy 中做对应解绑(off)处理。
8 mounted () {
9 this.$EventBus.$on('exitClassRoom',this.exitClassRoomHandle)
10 },
11 destroyed () {
12 this.$EventBus.$off('exitClassRoom',this.exitClassRoomHandle)
13 }

如果在mounted/created 钩子中使用了第三方库初始化,需要在beforeDestroy 中做对应销毁处理。
慎用keep-alive
当你用 keep-alive 包裹一个组件后,它的状态就会保留,因此就留在了内存里,切莫在整个路由页面上加上keep-alive。
一旦你使用了 keep-alive,那么你就可以访问另外两个生命周期钩子:activated和 deactivated。你需要在一个 keep-alive 组件被移除的时候,调用 deactivated 钩子进行清理或改变数据。

六、实际案例
教师工作台是单页应用,在来回切换路由时,页面耗用内存在一直上升。

通过全局搜索,发现部分事件没有解绑,部分添加代码如下

按照上面的1-5方法解决后,内存耗用还是在增长,怀疑不是组件内部的问题,从头开始读代码,发现问题:keep-alive绑定在了整个router-view上,而且代码写得很不友好,路由不同router-view也不同,代码如下:

 1 <keep-alive>
2 <el-container class="teacher-content" v-if="
3 ['teacherWorkbenchCourse', 'teacherWorkbenchReschedule'].indexOf(
4 this.$router.currentRoute.name
5 ) === -1
6 " key="teacher-content">
7 <el-header class="body-banner">
8
9 <Banner>
10 <DhProductLineSelect v-if="this.$router.currentRoute.name == 'teacherWorkbench'"></DhProductLineSelect>
11 <template v-if="
12 this.$router.currentRoute.name == 'personMaterial' ||
13 this.$router.currentRoute.name == 'contentCloud'
14 ">
15 <BreadCrumbMaterial></BreadCrumbMaterial>
16 </template>
17 <template v-if="
18 this.$router.currentRoute.name == 'searchResult' || this.lessonIsEdit">
19 <span class="router-go-back">
20 <img src />
21 </span>
22 </template>
23 <template v-if="!this.lessonIsEdit">
24 <BreadCrumbs />
25 </template>
26 </Banner>
27 </el-header>
28 <el-main class="body-main">
29
30 <router-view></router-view>
31 </el-main>
32 </el-container>
33
34 <router-view v-else> </router-view>
35 </keep-alive>
36 优化后:
37
38 <el-container
39 class="teacher-content"
40 key="teacher-content"
41 >
42 <el-header class="body-banner" :class="[headClass, marginLeft]">
43
44 <Banner>
45 <DhProductLineSelect v-if="this.$router.currentRoute.name == 'teacherWorkbench'||this.$router.currentRoute.name=='teacherWorkbenchCourse'"></DhProductLineSelect>
46 <template
47 v-if="
48 this.$router.currentRoute.name == 'personMaterial' ||
49 this.$router.currentRoute.name == 'contentCloud'
50 "
51 >
52 <BreadCrumbMaterial></BreadCrumbMaterial>
53 </template>
54 <template
55 v-if="
56 this.$router.currentRoute.name == 'searchResult' || this.lessonIsEdit"
57 >
58 <span class="router-go-back">
59 <img src />
60 </span>
61 </template>
62
63 <template v-if="!this.lessonIsEdit">
64 <BreadCrumbs/>
65 </template>
66 </Banner>
67 </el-header>
68 <el-main :class="['body-main',mainNoPadding]">
69 <router-view></router-view>
70 </el-main>
71 </el-container>

综上操作,内存耗用截图如下,无用内存大部分得到释放:

vue中处理过内存泄露处理方法的更多相关文章

  1. iPhone应用中如何避免内存泄露?

    如何有效控制iPhone内存管理的对象的所有权与引用计数和以及iPhone内存的自动释放与便捷方法.本文将介绍在iPhone应用中如何避免内存泄露.想了解“在iPhone应用中如何避免内存泄露”就必须 ...

  2. Java宝典(四)------Java中也存在内存泄露。

    --Java中会存在内存泄露吗? --如果你想当然的以为Java里有了垃圾回收机制就不会存在内存泄露,那你就错了. Java里也会存在内存泄露! 我们慢慢来分析. 所谓内存泄露就是指一个不再被程序使用 ...

  3. 对开发中常见的内存泄露,GDI泄露进行检测

    对开发中常见的内存泄露,GDI泄露进行检测 一.GDI泄露检测方法: 在软件测试阶段,可以通过procexp.exe 工具,或是通过任务管理器中选择GDI对象来查看软件GDI的对象是使用情况. 注意点 ...

  4. Flex内存泄露解决方法和内存释放优化原则

    Flex内存泄露解决方法和内存释放优化原则 你对Flex内存泄露的概念是否了解,这里和大家分享一下Flex内存释放优化原则和Flex内存泄露解决方法,希望本文的介绍能让你有所收获. Flex内存释放优 ...

  5. vue中使用echarts的两种方法

    在vue中使用echarts有两种方法一.第一种方法1.通过npm获取echarts npm install echarts --save 2.在vue项目中引入echarts 在 main.js 中 ...

  6. vue中push()和splice()的使用方法

    vue中push()和splice()的使用方法 push()使用 push() 方法可向数组的末尾添加一个或多个元素,并返回新的长度.注意:1. 新元素将添加在数组的末尾. 2.此方法改变数组的长度 ...

  7. Android开发中常见的内存泄露案例以及解决方法总结

    1.单例模式引起的内存泄露 由于单例模式的静态特性,使得它的生命周期和我们的应用一样长,如果让单例无限制的持有Activity的强引用就会导致内存泄漏如错误代码示例: public class Use ...

  8. java: web应用中不经意的内存泄露

    前面有一篇讲解如何在spring mvc web应用中一启动就执行某些逻辑,今天无意发现如果使用不当,很容易引起内存泄露,测试代码如下: 1.定义一个类App package com.cnblogs. ...

  9. 谈谈.NET中常见的内存泄露问题——GC、委托事件和弱引用

    其实吧,内存泄露一直是个令人头疼的问题,在带有GC的语言中这个情况得到了很大的好转,但是仍然可能会有问题.一.什么是内存泄露(memory leak)?内存泄露不是指内存坏了,也不是指内存没插稳漏出来 ...

随机推荐

  1. 《Java编程思想》学习笔记_多态

    多态 多态指一个行为产生多种状态,针对父类类型可接收其子类类型,最终执行的状态由具体子类确定,其不同子类可呈现出不同状态.例如人[父类]都会跑步[行为],但小孩[子类]跑步.成年人[子类]跑步.运动员 ...

  2. RPA应用场景-定点取数

    场景概述定点取数 所涉系统名称业务系统,Excel 人工操作(时间/次) 8 小时 所涉人工数量 2 操作频率实时 场景流程 1.从业务系统中拉取指定字段值的数据填入Excel: 2.将Excel每隔 ...

  3. Linux字符集和编码

    计算机内部,所有信息最终都是一个二进制值形式存放 字符集 字符集:charset是character set的简写,即二进制和字符的对应关系,不关注最终的存储形式 编码 字符集编码:encoding是 ...

  4. 业务可视化-让你的流程图"Run"起来

    前言 最近在研究业务可视化的问题,在日常的工作中,流程图和代码往往是分开管理的. 一个被维护多次的系统,到最后流程图和代码是否匹配这个都很难说. 于是一直有一个想法,让程序直接读流程图,根据流程图的配 ...

  5. 编译kubeadm使生成证书有效期为100年

    目录 问题 编译 检查结果 问题 当我使用kubeadm部署成功k8s集群时在想默认生成的证书有效期是多久,如下所示 /etc/kubernetes/pki/apiserver.crt #1年有效期 ...

  6. 总结vue 需要掌握的知识点

    使用的开发工具是webstorm,它是默认就安装好了vuejs插件,idea要使用的话,需要安装一下该插件 一.快速搭建项目vue-cli 脚手架(Vue2.0) 1.Vue CLI使用前提 –Nod ...

  7. 4-8 CS后台项目练习-2

    8. 类别管理--添加类别--持久层 8.1. 配置 续前日,无新增 8.2. 规划需要执行的SQL语句 续前日,无新增 8.3. 接口与抽象方法 此前需要执行的SQL语句大致是: select id ...

  8. 在.NET 6.0中配置WebHostBuilder

    大家好,我是张飞洪,感谢您的阅读,我会不定期和你分享学习心得,希望我的文章能成为你成长路上的垫脚石,让我们一起精进. 在阅读第4章"使用Kestrel配置和定制HTTPS"时,您可 ...

  9. python代码如何写的优雅?

    简介 在实际项目中,我们可能一开始为了完成功能而忽视了代码的整体质量,因此,使用一些高阶的函数或方法,能够更加使我们的代码更加优雅.废话不多说,现在马上开始. 使用enumerate方法替代range ...

  10. qbxt五一数学Day1

    目录 I. 基础知识 1. 带余除法(小学) 1. 定义 2. 性质 2. 最大公约数(gcd)/ 最小公倍数(lcm) 1. 定义 2. 性质 3. 高精度 II. 矩阵及其应用 1. 定义 2. ...