XingYun blog
  • JS基础

    • 图解js原型链
    • JS Event Loop
    • 对象的底层数据结构
    • 让你的JavaScript代码简单又高效
    • 函数参数按值传递
    • 判断数据类型
    • 浮点数精度问题和解决办法
    • 常用方法snippet
    • 实现Promise
    • 防抖和节流
    • 巧用sort排序
  • CSS && HTML

    • CSS也需要性能优化
    • class命名规范
    • em、px、rem、vh、vw 区别
    • CSS揭秘阅读笔记
  • 浏览器

    • 浏览器是如何渲染页面的
    • 重排和重绘
    • BOM浏览器对象模型
    • DOM事件
    • 浏览器存储
  • 数据结构

    • JS实现链表
    • JS实现栈与栈应用
    • JS实现常见排序
    • 哈夫曼编码
    • MD5算法
  • vue原理浅析

    • Vue虚拟dom与Diff算法
    • 前端打包文件的缓存机制
    • vue数组为什么不是响应式
    • v-for为什么不能用index做key
  • 前端工程化

    • 浏览器是如何渲染页面的
    • 前端打包需要gzip压缩吗
    • 前端打包文件的缓存机制
    • webpack loader和plugin
  • 轮子&&组件库

    • 实现水波浪进度球
  • 文字转语音mp3文件
  • 文件上传前后端实现
  • moment.js给定时间获取自然月、周的时间轴
  • 实现文件上传功能
  • 批量下载照片
  • leaflet改变坐标原点
  • 网络

    • 有了MAC地址 为什么还需要IP地址
    • 为什么IP地址老是变
    • 我们为什么需要IPV6
    • TCP与UDP
  • 计算机组成原理

    • ASCII、Unicode、UTF-8和UTF-16
  • VSCode

    • VSCode图片预览插件 Image preview
    • rsync:linux间的高效传输工具

XingYun

冲!
  • JS基础

    • 图解js原型链
    • JS Event Loop
    • 对象的底层数据结构
    • 让你的JavaScript代码简单又高效
    • 函数参数按值传递
    • 判断数据类型
    • 浮点数精度问题和解决办法
    • 常用方法snippet
    • 实现Promise
    • 防抖和节流
    • 巧用sort排序
  • CSS && HTML

    • CSS也需要性能优化
    • class命名规范
    • em、px、rem、vh、vw 区别
    • CSS揭秘阅读笔记
  • 浏览器

    • 浏览器是如何渲染页面的
    • 重排和重绘
    • BOM浏览器对象模型
    • DOM事件
    • 浏览器存储
  • 数据结构

    • JS实现链表
    • JS实现栈与栈应用
    • JS实现常见排序
    • 哈夫曼编码
    • MD5算法
  • vue原理浅析

    • Vue虚拟dom与Diff算法
    • 前端打包文件的缓存机制
    • vue数组为什么不是响应式
    • v-for为什么不能用index做key
  • 前端工程化

    • 浏览器是如何渲染页面的
    • 前端打包需要gzip压缩吗
    • 前端打包文件的缓存机制
    • webpack loader和plugin
  • 轮子&&组件库

    • 实现水波浪进度球
  • 文字转语音mp3文件
  • 文件上传前后端实现
  • moment.js给定时间获取自然月、周的时间轴
  • 实现文件上传功能
  • 批量下载照片
  • leaflet改变坐标原点
  • 网络

    • 有了MAC地址 为什么还需要IP地址
    • 为什么IP地址老是变
    • 我们为什么需要IPV6
    • TCP与UDP
  • 计算机组成原理

    • ASCII、Unicode、UTF-8和UTF-16
  • VSCode

    • VSCode图片预览插件 Image preview
    • rsync:linux间的高效传输工具
  • leaflet改变坐标原点
  • 补间动画gsap与tween
  • 文字转语音mp3文件
  • JavaScript引入
  • JavaScript高级程序设计阅读笔记
  • Javascript函数参数按值传递传递
  • JS防抖和节流
  • 手写JS常用方法
  • 手写Promise
  • JS Event Loop
  • 重排和重绘
  • em、px、rem、vh、vw 区别
  • css也需要性能优化
  • 图解 js 原型链
  • js函数参数按值传递
  • BOM浏览器对象模型
  • DOM
  • 事件
  • js对象数组sort按需排序
  • 文件上传功能技术选型和前后端实现
  • 前端图片处理
  • 让你的JavaScript代码简单又高效
  • BEM:class命名规范
  • 前端规范-CSS属性那么多(杂),怎么排序
  • TypeScript Tips
  • jsx
  • canvas基础
  • 前端日志
  • 浏览器存储
  • CSS世界阅读笔记
  • CSS揭秘阅读笔记
  • js变量命名常用规范词
  • 你不知道的JavaScript阅读笔记
  • js对象的底层数据结构
  • js判断数据类型
  • JS浮点数精度问题和解决办法
  • js作用域
  • js堆栈溢出和内存泄漏
    • 浏览器是如何渲染页面的
    • 疑难杂症和踩坑问题合集
    • 免费在线API收集
    • 原生JS实现Ajax请求
    • cookie、session、localStorage、sessionStorage的区别
    • Sass与Less
    • arrayBuffer、blob、file对象
    • TypeScript基础
    • 前端
    XingYun
    2022-02-03
    目录

    js堆栈溢出和内存泄漏

    概念:

    堆栈溢出: 程序内部函数的调用以及返回会不停的执行进栈和出栈的操作,栈空间是有限的,一旦调用即进栈过多就会导致栈满

    内存泄漏: 申请的内存执行完之后没有及时的清理和销毁,占用空闲内存,既不能使用也不能回收

    # 一、堆栈溢出

    JS 中的数据存储分为栈和堆,栈遵循先进后出的原则,所以程序从栈底开始计算,程序内部函数的调用以及返回会不停的执行进栈和出栈的操作,一旦调用即进栈过多就会导致栈满。

    这种情况常见于递归调用

    # 1.什么是函数调用

    var a = 2
    function add() {
      var b = 10
      return a + b
    }
    add()
    
    1
    2
    3
    4
    5
    6

    我用这段简单的代码来解释下函数调用的过程。

    在执行到函数add()之前,JavaScript引擎会为上面这段代码创建全局执行上下文,包含了声明的函数和变量,你可以参考下图:

    # 2.函数执行过程

    生成可执行代码之后,JS引擎开始顺序执行代码,执行到add这里时,JS引擎判断出这里是函数调用,然后执行下面操作:

    1. 从全局上下文中,取出add函数代码
    2. 对add函数的这段代码进行编译(创建该函数的执行上下文环境和可执行代码)
    3. 执行add函数,输出结果

    在执行add函数时,会存在两个执行上下文,一个是全局执行上下文,一个是add函数的执行上下文。

    那么JS引擎是怎么管理多个执行上下文的呢,JS引擎是通过栈来管理这些执行上下文的。

    # 3.一个递归造成栈溢出的例子

    下面是一个递归爆炸的例子:

    function test(n) {
      if (n === 0) return true
      return test(n - 1)
    }
    
    1
    2
    3
    4

    当 n 为 100 时,运行时很快输出 true,

    当 n 为 10000000 时,会抛出错误 VM9657:2 Uncaught RangeError: Maximum call stack size exceeded

    当JavaScript引擎开始执行这段代码时,它首先调用函数test,并创建执行上下文,压入栈中;

    因为这个函数是递归的,所以它会一直创建新的函数执行上下文,并反复将其压入栈中,直到 n<=1 ,但栈是有容量限制的,超过最大数量后就会出现了栈溢出的错误。

    理解了栈溢出原因后,你就可以使用一些方法来避免或者解决栈溢出的问题,比如把递归调用的形式改造成尾递归

    # 4.总结

    1. 每调用一个函数,JavaScript引擎会为其创建执行上下文,并把该执行上下文压入调用栈,然后JavaScript引擎开始执行函数代码。
    2. 如果在一个函数A中调用了另外一个函数B,那么JavaScript引擎会为B函数创建执行上下文,并将B函数的执行上下文压入栈顶。
    3. 当前函数执行完毕后,JavaScript引擎会将该函数的执行上下文pop出栈。
    4. 当分配的调用栈空间被占满时,会引发“堆栈溢出”问题。

    # 二、内存泄露

    申请的内存执行完之后没有及时的清理和销毁,占用空闲内存,既不能使用也不能回收。 几种会导致内存泄露的情况:

    1. 全局变量满天飞
    2. 没销毁的计时器或回调函数
    3. 没释放的 DOM 的引用
    4. 闭包使用不当

    # 解决方式

    1. 减少不必要的全局变量,使用 Javascript 严格模式来避免创建意外的全局变量

    2. 每次使用 setTimeout 和 setInterval, 都要注意在适当的时机手动销毁

    3. 使用完数据之后,及时解除引用(闭包中的变量、 DOM 引用)

    bad

    <div id="root">
      <div class="child">child</div>
      <button>remove</button>
    </div>
    <script>
      let btn = document.querySelector('button')
      let child = document.querySelector('.child')
      let root = document.querySelector('#root')
      btn.addEventListener('click', function() {
        root.removeChild(child) //虽然该.child节点确实从dom中被移除了,但全局变量child仍然对该节点有引用,导致该节点的内存一直无法释放
      })
    </script>
    
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12

    good

    <div id="root">
      <div class="child">child</div>
      <button>remove</button>
    </div>
    <script>
      let btn = document.querySelector('button')
    
      btn.addEventListener('click', function() {
        let child = document.querySelector('.child')
        let root = document.querySelector('#root')
        //当移除节点并退出回调函数的执行上下文后会自动清除对该节点的引用
        root.removeChild(child)
      })
    </script>
    
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    #JavaScript
    上次更新: 2023/04/05, 09:41:10
    js作用域
    浏览器是如何渲染页面的

    ← js作用域 浏览器是如何渲染页面的→

    最近更新
    01
    JavaScript-test
    07-20
    02
    二维码的原理
    07-20
    03
    利用ChatGPT优化代码
    07-20
    更多文章>
    Theme by Vdoing | Copyright © 2021-2023 XingYun | MIT License
    • 跟随系统
    • 浅色模式
    • 深色模式
    • 阅读模式