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

JavaScript 性能优化:从入门到实战

在当今快节奏的互联网时代,用户对网页和应用的加载速度与响应性能要求越来越高。JavaScript 作为网页交互的核心语言,其性能表现直接影响用户体验。本文将用简单易懂的语言,带你了解 JavaScript 性能优化的实用技巧,帮助你的代码跑得更快、更流畅。

一、减少全局变量的使用

在 JavaScript 中,全局变量会挂载在全局对象(浏览器环境下是window)上。频繁访问全局变量时,JavaScript 引擎需要在全局作用域中查找变量,相比局部变量,查找效率较低。例如:

// 不好的写法var globalVar = 10;function someFunction() {console.log(globalVar);}// 好的写法function someFunction() {var localVar = 10;console.log(localVar);}

将变量声明在函数内部,形成局部变量,引擎查找局部变量的速度更快,能提升代码执行效率。同时,过多的全局变量还可能导致命名冲突,使代码难以维护,减少全局变量使用一举多得。

二、优化循环操作

循环是 JavaScript 中常用的代码结构,但如果使用不当,很容易造成性能问题。以遍历数组为例:

// 普通for循环var arr = new Array(10000).fill(0);for (var i = 0; i < arr.length; i++) {// 执行操作}// 优化后的for循环var arr = new Array(10000).fill(0);var len = arr.length;for (var i = 0; i < len; i++) {// 执行操作}

在普通 for 循环中,每次循环都需要获取arr.length的值,而数组长度在循环过程中一般不会改变,反复获取就造成了不必要的性能开销。优化后的代码将arr.length的值缓存起来,避免了重复计算,能有效提升循环执行效率。

另外,在遍历数组时,也可以根据实际需求选择更合适的方法。比如,forEach、map、filter等数组方法虽然使用起来简洁方便,但它们内部会有一些额外的函数调用开销。如果对性能要求较高,且不需要使用这些方法的特殊功能,使用普通for循环可能是更好的选择。

三、事件委托

在网页开发中,常常需要为大量的 DOM 元素添加事件监听器。如果为每个元素单独添加监听器,会占用大量内存,影响性能。事件委托就是一种很好的优化方案。

事件委托利用了事件冒泡的原理。当一个元素触发事件时,该事件会从触发元素开始,逐级向上冒泡到 DOM 树的根节点。我们可以将事件监听器添加到这些元素的共同祖先元素上,通过判断事件源来决定是否执行相应的操作。

例如,为列表中的每个li元素添加点击事件:

<ul id="list"><li>Item 1</li><li>Item 2</li><li>Item 3</li></ul>// 不好的写法var lis = document.querySelectorAll('#list li');lis.forEach(function(li) {li.addEventListener('click', function() {console.log('Clicked!');});});// 好的写法 - 事件委托var list = document.getElementById('list');list.addEventListener('click', function(event) {if (event.target.tagName === 'LI') {console.log('Clicked!');}});

使用事件委托,只需为ul元素添加一个事件监听器,无论列表中有多少个li元素,都能高效处理点击事件,大大减少内存占用。

四、避免频繁操作 DOM

DOM 操作是 JavaScript 性能的一大瓶颈。因为 DOM 操作涉及到浏览器的渲染引擎,每次对 DOM 进行修改,浏览器都需要重新计算样式、布局,并进行重绘和回流,这是非常消耗性能的。

例如,批量修改元素的样式:

// 不好的写法var div = document.getElementById('myDiv');div.style.color ='red';div.style.fontSize = '16px';div.style.margin = '10px';// 好的写法var div = document.getElementById('myDiv');div.className = 'new-style';

第一种写法每次修改style属性都会触发浏览器的重绘和回流。而第二种写法,事先在 CSS 中定义好.new-style样式类,通过修改className一次性应用所有样式,减少了 DOM 操作次数,提升了性能。

如果需要对 DOM 进行大量修改,可以先将元素从 DOM 树中移除(使用removeChild或detach方法,不同库和环境略有差异),在内存中完成修改后,再将其添加回 DOM 树,这样可以避免多次触发重绘和回流。

五、内存管理

JavaScript 具有自动垃圾回收机制,但如果使用不当,仍会造成内存泄漏,影响性能。常见的内存泄漏场景及解决方法如下:

  1. 意外的全局变量:使用严格模式('use strict';),可以避免意外创建全局变量。在严格模式下,未声明的变量赋值会报错,帮助我们及时发现问题。
  1. 未清理的定时器和事件监听器:在不需要定时器或事件监听器时,一定要调用clearTimeout、clearInterval清除定时器,使用removeEventListener移除事件监听器。例如:
var timer = setInterval(function() {// 执行操作}, 1000);// 不再需要时清除定时器clearInterval(timer);var element = document.getElementById('myElement');var handler = function() {// 处理函数};element.addEventListener('click', handler);// 移除事件监听器element.removeEventListener('click', handler);

  1. 闭包引起的内存泄漏:闭包会持有外部作用域的变量,如果使用不当,这些变量无法被垃圾回收。在闭包使用完毕后,及时将相关变量设置为null,帮助垃圾回收机制回收内存。

六、代码压缩与合并

在项目上线前,对 JavaScript 代码进行压缩和合并是常见的优化手段。压缩可以去除代码中的空格、注释等冗余内容,减小文件体积,加快加载速度。合并则是将多个 JavaScript 文件合并成一个,减少浏览器的 HTTP 请求次数。

可以使用工具如UglifyJS、Terser进行代码压缩,使用Webpack、Rollup等构建工具实现代码的合并与打包。

通过以上这些优化技巧,我们可以显著提升 JavaScript 代码的性能。在实际开发中,要根据具体的应用场景,灵活运用这些方法,并结合浏览器的开发者工具(如 Chrome DevTools 的性能分析功能),对代码进行性能监测和调优,打造出高效、流畅的网页和应用。

相关文章:

JavaScript 性能优化:从入门到实战

在当今快节奏的互联网时代&#xff0c;用户对网页和应用的加载速度与响应性能要求越来越高。JavaScript 作为网页交互的核心语言&#xff0c;其性能表现直接影响用户体验。本文将用简单易懂的语言&#xff0c;带你了解 JavaScript 性能优化的实用技巧&#xff0c;帮助你的代码跑…...

启动metastore时报错MetaException(message:Version information not found in metastore

把hdfs清空重新安装了一下&#xff0c;hive的mysql元数据库删除掉之后重建之后一直启动报错 metastore.RetryingHMSHandler (RetryingHMSHandler.java:<init>(83)) - HMSHandler Fatal error: MetaException(message:Version information not found in metastore.) 后来…...

Spring 中 @Value 注解多实例配置方案详解

引言 在使用 Spring 框架进行开发时&#xff0c;我们经常会使用 Value 注解来注入配置值。然而&#xff0c;当我们需要创建同一个类的多个实例&#xff0c;并且每个实例需要使用不同的配置值时&#xff0c;直接在类中使用 Value 注解就会遇到问题。本文将深入探讨这个问题&…...

MyBatisPlus(1):快速入门

我们知道&#xff0c;MyBatis是一个优秀的操作数据库的持久层框架&#xff08;优秀持久层框架——MyBatis&#xff09;&#xff0c;其基于底层的JDBC进行高度封装&#xff0c;极大的简化了开发。但是对于单表操作而言&#xff0c;我们需要重复地编写简单的CRUD语句。这其实是不…...

京东热点缓存探测系统JDhotkey架构剖析

热点探测使用场景 MySQL 中被频繁访问的数据 &#xff0c;如热门商品的主键 IdRedis 缓存中被密集访问的 Key&#xff0c;如热门商品的详情需要 get goods$Id恶意攻击或机器人爬虫的请求信息&#xff0c;如特定标识的 userId、机器 IP频繁被访问的接口地址&#xff0c;如获取用…...

多国金融市场数据对接指南(印度、印尼、韩国)

一、StockTV多国数据对接概述 StockTV提供统一的API接口规范对接全球金融市场数据&#xff0c;本文重点介绍印度&#xff08;国家ID:14&#xff09;、印尼&#xff08;国家ID:42&#xff09;、韩国&#xff08;国家ID:xx&#xff09;的股票市场对接方案。 共同特性 统一认证…...

【Elasticsearch】ILM(Index Lifecycle Management)策略详解

ILM&#xff08;Index Lifecycle Management&#xff09;策略详解 1.什么是 ILM 策略&#xff1f;2.ILM 解决的核心业务问题3.ILM 生命周期阶段3.1 Hot&#xff08;热阶段&#xff09;3.2 Warm&#xff08;温阶段&#xff09;3.3 Cold&#xff08;冷阶段&#xff09;3.4 Delete…...

linux 后记

Linux Server 下载一个Server的版本&#xff0c;就是那种只有命令行的 学会这个就可以去租一个aliyun服务器&#xff0c;挺便宜的 如果在aliyun买服务器的话就不用管镜像源 但是如果是自己的虚拟机就必须设置镜像源&#xff0c;上网搜索阿里的镜像源&#xff0c;然后手动输入&…...

【笔记】在 MSYS2 MINGW64 环境中安装构建工具链(CMake、GCC、Make)

&#x1f4dd; 在 MSYS2 MINGW64 环境中安装构建工具链&#xff08;CMake、GCC、Make&#xff09; ✅ 目标说明 记录在 MSYS2 的 MINGW64 工具链环境中&#xff0c;成功安装用于 C/C 构建的常用开发工具。 包括&#xff1a; GCC 编译器Make 构建系统CMake 跨平台构建工具基础开…...

PyTorch -TensorBoard的使用 (一)

设置环境 新建python文件 .py 安装Tensorboard 在终端进行安装 显示安装成功 两个logs&#xff0c;出现这种情况怎么解决 所有的logs文件删掉delete&#xff0c;重新运行 add_image 不满足要求 Opencv-numpy 安装Opencv add_image 用法示例 &#xff08;500&#xff0c;375&am…...

Redis最佳实践——性能优化技巧之数据结构选择

Redis在电商应用中的数据结构选择与性能优化技巧 一、电商核心场景与数据结构选型矩阵 应用场景推荐数据结构内存占用读写复杂度典型操作商品详情缓存Hash低O(1)HGETALL, HMSET购物车管理Hash中O(1)HINCRBY, HDEL用户会话管理Hash低O(1)HSETEX, HGET商品分类目录Sorted Set高O…...

网络安全方向在校生有哪些证书适合考取?

工作7年得出结论&#xff1a;网络安全&#xff0c;考任何证书都没有用&#xff0c;实力才是根本。我是2021年考的 CISSP&#xff0c;报了培训班&#xff0c;花了1万一千块钱&#xff0c;签的保障班还是服务班不记得了&#xff0c;大概意思就是你放心去考&#xff0c;考不过可以…...

从0开始学习R语言--Day14--贝叶斯统计与结构方程模型

贝叶斯统计 在很多时候&#xff0c;我们经常会看到在统计分析中出现很多反直觉的结论&#xff0c;比如假如有一种病&#xff0c;人群中的患病率为1%&#xff0c;患者真患病时&#xff0c;检测结果为阳性的概率是99%&#xff0c;如果没有&#xff0c;则检测结果为阳性的概率是5…...

02-BTC-密码学原理 对hash算法如果出现漏洞的思考

如果比特币中某个哈希函数的抗碰撞性出现了漏洞怎么办&#xff0c;怎么补救&#xff1f; 答&#xff1a;&#xff08;1&#xff09;攻击场景&#xff1a; 伪造交易&#xff1a;攻击者可构造两个不同的交易&#xff08;如正常交易和恶意双花交易&#xff09;具有相同的TxID&…...

[Python] 如何使用 Python 调用 Dify 工作流服务实现自动化翻译

在实际项目中,自动化工作流服务可以大大简化复杂任务的处理流程。本文将介绍如何通过 Python 脚本调用 Dify 提供的工作流 API,实现文本翻译的自动化操作。该流程包括设置 API 接口、构造请求体并处理返回结果。 一、背景介绍:什么是 Dify 工作流服务? Dify 是一款支持多种…...

分布式微服务系统架构第142集:全栈开发

加群联系作者vx&#xff1a;xiaoda0423 仓库地址&#xff1a;https://webvueblog.github.io/JavaPlusDoc/ https://1024bat.cn/ https://github.com/webVueBlog/fastapi_plus https://webvueblog.github.io/JavaPlusDoc/ /*** 本地启动解决跨域问题* 打包发布请注释该类&#…...

PTA-根据已有类Worker,使用LinkedList编写一个WorkerList类,实现计算所有工人总工资的功能。

目录 1.问题描述 2.函数接口定义&#xff1a; 3.裁判测试程序样例&#xff1a; 4.输入和输出样例 输入样例&#xff1a; 输出样例&#xff1a; 5.实现代码 1.问题描述 Main类&#xff1a;在main方法中&#xff0c;调用constructWorkerList方法构建一个Worker对象链表…...

文档整合自动化

主要功能是按照JSON文件&#xff08;Sort.json&#xff09;中指定的顺序合并多个Word文档&#xff08;.docx&#xff09;&#xff0c;并清除文档中的所有超链接。最终输出合并后的文档名为"sorted_按章节顺序.docx"。 主要分为几个部分&#xff1a; 初始化配置 定…...

微软markitdown PDF/WORD/HTML文档转Markdown格式软件整合包下载

本次和大家分享另一个微软发布的非常热门的文件文档转Markdown格式文档的软件markitdown&#xff0c;软件可以将PDF&#xff0c;word&#xff0c;ppt&#xff0c;Excel等十几种格式文档转换为markdown格式文档&#xff0c;我基于当前最新0.1.2版本制作了免安装一键启动整合包。…...

科普:Linux `su` 切换用户后出现 `$` 提示符,如何排查和解决?

科普&#xff1a;Linux su 切换用户后出现 $ 提示符&#xff0c;如何排查和解决&#xff1f; 在 Linux 系统管理中&#xff0c;su&#xff08;Switch User&#xff09;命令用于切换用户身份。正常情况下&#xff0c;从 root 切换到普通用户时&#xff0c;提示符会从 # 变成 $&…...

BayesFlow:基于神经网络的摊销贝叶斯推断框架

贝叶斯推断为不确定性条件下的推理、复杂系统建模以及基于观测数据的预测提供了严谨且功能强大的理论框架。尽管贝叶斯建模在理论上具有优雅性&#xff0c;但在实际应用中经常面临显著的计算挑战&#xff1a;后验分布通常缺乏解析解&#xff0c;模型验证和比较需要进行重复的推…...

NodeJS全栈开发面试题讲解——P9性能优化(Node.js 高级)

✅ 9.1 Node.js 的性能瓶颈一般出在哪&#xff1f;如何排查&#xff1f; Node.js 单线程 异步模型&#xff0c;瓶颈常出现在&#xff1a; 阻塞操作&#xff08;如&#xff1a;同步 I/O、CPU 密集型计算&#xff09; 数据库慢查询 / 索引失效 外部接口慢响应 大量并发请求导…...

NVMe IP现状扫盲

SSD优势 与机械硬盘&#xff08;Hard Disk Driver, HDD&#xff09;相比&#xff0c;基于Flash的SSD具有更快的数据随机访问速度、更快的传输速率和更低的功耗优势&#xff0c;已经被广泛应用于各种计算领域和存储系统。SSD最初遵循为HDD设计的现有主机接口协议&#xff0c;例…...

5G-A时代与p2p

5G-A时代正在走来&#xff0c;那么对P2P的影响有多大。 5G-A作为5G向6G过渡的关键技术&#xff0c;将数据下载速率从千兆提升至万兆&#xff0c;上行速率从百兆提升至千兆&#xff0c;时延降至毫秒级。这种网络性能的跨越式提升&#xff0c;为P2P提供了更强大的底层支撑&#x…...

基于FPGA的DES加解密系统verilog实现,包含testbench和开发板硬件测试

目录 1.课题概述 2.系统测试效果 3.核心程序与模型 4.系统原理简介 5.完整工程文件 1.课题概述 基于FPGA的DES加解密系统verilog实现,包含testbench和开发板硬件测试。输入待加密数据&#xff0c;密钥&#xff0c;输出加密数据&#xff0c;然后通过解密模块输出解密后的原…...

基于生产-消费模式,使用Channel进行文件传输(Tcp方式)

Client端&#xff1a; #region 多文件传输 public class FileMetadata {public string FileName { get; set; }public long FileSize { get; set; } }class Program {const int PORT 8888;const int BUFFER_SIZE 60 * 1024 * 1024;//15s-50 25s-64 33s-32 27s-50 31s-40 25…...

tortoisegit 使用rebase修改历史提交

在 TortoiseGit 中使用 rebase 修改历史提交&#xff08;如修改提交信息、合并提交或删除提交&#xff09;的步骤如下&#xff1a; --- ### **一、修改最近一次提交** 1. **操作**&#xff1a; - 右键项目 → **TortoiseGit** → **提交(C)** - 勾选 **"Amend…...

Python----目标检测(《用于精确目标检测和语义分割的丰富特征层次结构》和R-CNN)

一、《用于精确目标检测和语义分割的丰富特征层次结构》 1.1、基本信息 原文标题&#xff1a;Rich feature hierarchies for accurate object detection and semantic segmentation 中文译名&#xff1a;用于精确目标检测与语义分割的丰富特征层次结构 版本&#xff1a;第5版技…...

Ansible 进阶 - Roles 与 Inventory 的高效组织

Ansible 进阶 - Roles 与 Inventory 的高效组织 如果说 Playbook 是一份完整的“菜谱”,那么 Role (角色) 就可以被看作是制作这道菜(或一桌菜)所需的标准化“备料包”或“半成品组件”。例如,我们可以有一个“Nginx Web 服务器安装配置 Role”、“MySQL 数据库基础设置 Ro…...

极简以太彩光网络解决方案4.0正式发布,“彩光”重构园区网络极简之道

5月28日下午,锐捷网络在京举办以“光,本该如此‘简单’”为主题的发布会,正式发布极简以太彩光网络解决方案4.0。作为“彩光”方案的全新进化版本,极简以太彩光4.0从用户需求出发,聚焦场景洞察,开启了一场从底层基因出发的极简革命,通过架构、部署、运维等多维度的创新升级,以强…...