通过这个简单的技巧让我们的 JavaScript 代码变得异常快
通过这个简单的技巧让我们的 JavaScript 代码变得异常快
秘诀:了解JavaScript 虚拟机(VM)的内部工作原理。
首先,我们来谈谈像 V8
这样的JavaScript
虚拟机(VM
)。可以把它想象成我们的操作的大脑 —— 它将我们简洁的代码变成计算机可以理解和执行的东西。
好的、坏的代码
让我们深入研究一些代码示例,看看好的、坏的和执行快的代码。
缓慢的 JavaScript
示例:
function addProperty(obj, propName, value) {obj[propName] = value; // 这会改变对象的形状
}
const responseObject = { user1: 1, user2: 2 };
addProperty(responseObject, 'user3', 3); // 添加新的属性
是什么让它变慢?
形状更改:每次调用 addProperty
函数时,都会向对象添加一个新属性。这会改变对象的“形状”,即它包含的键变了,这反过来又会颠覆 JavaScript
引擎的优化。
添加或删除属性时,引擎可能必须丢弃以前的优化信息并重新开始。这种“形状变化”就是操作缓慢的原因。
快速 JavaScript
示例:
function createObject ( a, b, c ) {
// 对象的形状是固定的并且VM可以预测return { a, b, c };
}
const dataObject = createObject ( 212,2344,43545);
是什么让它如此快速?
可预测的形状:
该对象是使用一组固定的属性创建的。创建后没有任何变化,更容易引擎优化。
隐藏类重用:
由于每次调用 createObject
时对象的形状都是一致的,因此 JavaScript
引擎可以重用为此形状创建的隐藏类。这种重用允许非常快速的属性访问,因为引擎确切地知道每个属性在内存中的位置。
为什么对象形状很重要:
当我们访问对象的属性时,引擎不想搜索所有属性来找到它。相反,它想要直接访问该属性在内存中的位置。如果对象的形状已知,引擎可以记住每个属性所在的位置(这称为“内联缓存”)。但是,如果形状发生变化(如上面的缓慢示例),引擎必须“重新学习”属性位置,这要慢得多。
为了获得最佳性能,特别是在频繁访问属性的代码关键部分,最好:
- 创建对象时初始化所有属性:即使某些属性最初可能未定义。
- 避免添加或删除属性:这可以保持对象的形状稳定。
- 尽可能重用对象形状:创建始终生成具有相同属性集的对象的工厂函数。
通过遵循这些实践,我们可以帮助 JavaScript
引擎优化我们的代码,从而加快执行速度。
常见的用例
当处理来自外部源的对象(例如 API
响应或 DOM
元素)时,在使用这些对象之前将它们规范化为一致的形状对性能有益。这允许 JavaScript
引擎优化对这些对象的访问,因为形状(对象内的所有键)是可预测的并且不会改变。当我们频繁读取对象时,这种做法尤其有价值。
让我们来看下面两个常见的示例
通过 API
获取用户信息
慢速版本:
在慢速版本中,属性被一一添加到对象中,这可能会导致 JavaScript
引擎由于形状变化而取消对对象的访问优化。
function fetchUserProfile(url) {fetch(url).then(response => response.json()).then(user => {const userProfile = {};if (user.name) {userProfile.name = user.name;}if (user.age) {userProfile.age = user.age;}if (user.email) {userProfile.email = user.email;}// ...处理更多的属性return userProfile;});
}
快速版本:
在快速版本中,我们从一开始就创建一个具有已知形状的对象,即使某些属性可能未定义。这种一致性允许 JavaScript
引擎优化属性访问。
function fetchUserProfile(url) {return fetch(url).then(response => response.json()).then(user => {// 先定义对象中包含的所有的属性const userProfile = {name: user.name || undefined,age: user.age || undefined,email: user.email || undefined,// ... 初始化更多属性};return userProfile;});
}
在快速版本中,即使用户对象不具有我们分配给 userProfile
的所有属性,我们仍然使用其相应的值或未定义的值来定义我们期望的所有键。这样,userProfile
的形状保持一致,这有利于稍后访问其属性时的性能。
这种做法对于性能关键型应用程序至关重要,优化可以极大地提高执行速度。
如果上面的例子让我们想起了什么,那是因为这个模式看起来像工厂模式,它遵循类似于工厂函数的原则,通过创建一个具有预定义形状的对象,但它并不完全是这样。在 JavaScript
中,工厂模式通常涉及构造并返回新对象的专用函数。当创建过程复杂或需要执行一些额外的设置工作时,工厂函数特别有用。
使用工厂模式
在给定的快速示例中,我们看到了一种创建具有一致形状的对象的方法。为了使其与工厂模式更加一致,我们可以将对象创建封装在专用函数中,如下所示:
function createUserProfile(name, age, email) {
// 通过工厂模式创建对象return {name: name || undefined,age: age || undefined,email: email || undefined,// ...};}function fetchUserProfile(url) {return fetch(url).then(response => response.json()).then(user => {return createUserProfile(user.name, user.age, user.email);});
}
在这个版本中,createUserProfile
是一个工厂函数,总是创建具有相同形状的对象,这有利于优化。fetchUserProfile
函数使用此工厂创建一个新的 userProfile
对象。
使用DOM
现在让我们讨论另一个常见的示例,在使用 DOM
时,我们经常需要从 HTML
元素读取信息,然后在应用程序中使用这些数据。保持对象形状一致对于性能非常重要,尤其是当我们重复执行这些操作时。
下面的示例演示了对象形状发生变化的慢速代码示例,以及对象形状可预测且一致的快速方法。
慢速代码示例
function getUserData() {const userObject = {};const userName = document.querySelector('#input-name');if (nameElement) {userObject.name = nameElement.textContent;}const userAge = document.querySelector('#input-age');if (ageElement) {userObject.age = parseInt(ageElement.textContent);}// 每次调用此函数时,它可能会也可能不会添加新属性// 这可能会导致对象形状发生变化return userObject;
}
快速代码示例
function createUserData(name = undefined, age = undefined) {
// 始终返回具有相同形状的对象的工厂函数return { name, age };
}
function getUserData() {const userName = document.querySelector('#input-name');const userAge = parseInt(document.querySelector('#input-age')?.textContent);// 无论元素是否存在,对象的形状都是一致的return createUserData(userName?.textContent, Number.isNaN(userAge) ? undefined : userAge);
}
在上面代码中,createUserData
工厂函数确保返回的对象始终具有相同的形状,这有利于 JavaScript
引擎的优化过程。getUserData
函数使用此工厂函数来创建配置文件数据对象,并通过提供 undefined
作为默认值来处理丢失的 DOM
元素,从而维护对象的形状。
通过使用可选链接运算符 (?.
) 和空合并运算符 (??
),我们可以进一步细化该函数以处理 DOM
元素可能不存在的情况:
function getUserData() {const name = document.querySelector('#input-name')?.textContent ?? undefined;const ageText = document.querySelector('#input-age')?.textContent ?? undefined;const age = ageText ? parseInt(ageText) : undefined;// 对象的形状一致return createUserData(name, age);
}
这种方法可以确保对象的形状保持不变,即使在 DOM
中找不到某些元素,这在动态 Web
应用程序中很常见,因为有时元素尚未渲染或元素渲染顺序不正确。
相关文章:
通过这个简单的技巧让我们的 JavaScript 代码变得异常快
通过这个简单的技巧让我们的 JavaScript 代码变得异常快 秘诀:了解JavaScript 虚拟机(VM)的内部工作原理。 首先,我们来谈谈像 V8 这样的JavaScript 虚拟机(VM)。可以把它想象成我们的操作的大脑 —— 它将我们简洁的代码变成计算机可以理解和执行的东…...

vue怎么实现国际化? vue-i18n 插件实现国际化,支持切换不同语言
依赖的文档开始 | Vue I18n 一、安装 npm install vue-i18n 如果在一个模块系统中使用它,你必须通过 Vue.use() 明确地安装 vue-i18n: import Vue from vue import VueI18n from vue-i18nVue.use(VueI18n)二、使用 在 src 下创建 lang 文件夹 1.准…...

rabbit MQ的延迟队列处理模型示例(基于SpringBoot延时插件实现)
rabbitMQ安装插件rabbitmq-delayed-message-exchange 交换机由此type 表示组件安装成功 生产者发送消息时设置延迟值 消息在交换机滞纳至指定延迟后,进入队列,被消费者消费。 组件注解类: package com.esint.configs;import org.springfra…...

虽不想承认,但这就是CSGO游戏搬砖行业的现状
CSGO搬砖日常出货更新 其实整个搬砖市场,现在已经变得乌烟瘴气,散发着“恶臭”。我个人非常鄙视那些虚有其表,大小通吃的做法,那些甚至连搬砖数据都看不懂的人,也出来吹嘘着“实力强大,经验丰富”。这个世界…...

想问问各位大佬,网络安全这个专业普通人学习会有前景吗?
网络安全是一个非常广泛的领域,涉及到许多不同的岗位。这些岗位包括安全服务、安全运维、渗透测试、web安全、安全开发和安全售前等。每个岗位都有自己的要求和特点,您可以根据自己的兴趣和能力来选择最适合您的岗位。 渗透测试/Web安全工程师主要负责模…...

uniapp IOS从打包到上架流程(详细简单) 原创
1.登入苹果开发者网站,打开App Store Connect 2.新App的创建 点击我的App可以进入App管理界面,在右上角点击➕新建App 即可创建新的App,如下图: 3.app基本信息填写 新建完App后,需要填写App的基本信息&…...
React Native项目接入Sentry指南
本文主要介绍React Native项目接入Sentry流程,以及遇到的一些注意点,方便大家去解决和处理,如果在接入过程中,遇到任何问题可以在评论区留言,我将根据自己的接入经验给出一些解决方案和建议。 1, 安装sentry sdk 我们可以在项目中执行如下命令来安装sentry,命令如下: …...

首批!创邻科技入选《图数据库金融应用场景优秀案例》
11月11日,“全球金融科技中心网络年会”在第三届全球金融科技大会暨第五届成方金融科技论坛上成功在京举办。会上,北京前沿金融监管科技研究院发布了基于国际标准组织——国际关联数据基准委员会(LDBC)的《图数据库金融应用场景优…...

WPF树形控件TreeView使用介绍
WPF 中的 TreeView 控件用于显示层次结构数据。它是由可展开和可折叠的 TreeViewItem 节点组成的,这些节点可以无限嵌套以表示数据的层次。 TreeView 基本用法 例如实现下图的效果: xaml代码如下: <Window x:Class"TreeView01.Mai…...

Django 模型和Admin站点管理(三)
一、定义模型 (1) 创建模型类,必须要继承自 models.Model from django.db import models# Create your models here. #设计数据库 #创建模型 class UserModel(models.Model):namemodels.CharField(max_length30) #对应于SQL name varchar(30…...

JVMj之console Java监视与管理控制台
jconsole Java监视与管理控制台 1、jconsole介绍 jconsole (java monitoring and management console)是一款基于JMX (Java Management Extensions) 的可视化监视和管理工具。 2、启动jconsole 1、在linux和windwos下通过jconsole启动即可。 2、然后会自动搜索本机运行的…...

Android进阶知识:ANR的定位与解决
1、前言 ANR对于Android开发者来说一定不会陌生,从刚开始学习Android时的一不注意就ANR,到后来知道主线程不能进行耗时操作注意到这点后,程序出现ANR的情况就大大减少了,甚至于消失了。那么真的是只要在主线程做耗时操作就会产生…...

基于SSM的老年公寓信息管理(有报告)。Javaee项目
演示视频: 基于SSM的老年公寓信息管理(有报告)。Javaee项目 项目介绍: 采用M(model)V(view)C(controller)三层体系结构,通过Spring SpringMvc …...

阿里云 ACK 新升级,打造智算时代的现代化应用平台
云布道师 今天,能想到的或是想不到的领域,对容器和 Kubernetes 的需求都居高不减,使这项技术正在真正走向无处不在。 在 2023 云栖大会上,阿里云云原生产品线容器服务负责人易立关于容器服务 ACK 在本届亚运会上应用的介绍&#…...

CodeWhisperer 一款好玩的 AI 插件
忙里抽闲,今天试了试 CodeWhisperer 这款插件,我是在 IDEA 中做的测试,下面是我的一些使用感想: 安装 CodeWhisperer 插件:在 IntelliJ IDEA 中,可以通过插件管理器安装 CodeWhisperer 插件,然…...

大模型生态新篇章:以AI Agent为引,助企业创新应用落地
文 | 智能相对论 作者 | 沈浪 以聊天机器人、虚拟助手、智能客服等为代表的对话式人工智能 (Conversational AI Agents ) 在具体服务场景中的应用已经十分普遍。今年以来,随着大模型技术的爆发与加持,对话式AI被市场赋予了更高的期望。 “所有行业都值…...

什么是网络安全工程师,你想知道的都在这里!
随着互联网的发展和大数据时代的到来,网络已经日渐深入到我们生活、工作中的方方面面,社会信息化和信息网络化,突破了应用信息在时间和空间上的障碍,使信息的价值不断提高。但是,与此同时,网页篡改、计算机…...

uniapp开发的微信小程序进行代码质量控制,分包+压缩js+组件按需注入等
小程序代码分包的操作请看另外一篇文章:uniapp分包优化,包括分包路由跳转规则-CSDN博客 JS文件压缩:在工具「详情」-「本地设置」中开启「上传代码时自动压缩脚本文件」的设置 代码包:组件 > 启用组件按需注入解决办法 在小程…...

GD32替换STM32使用HAL库开发问题
GD32HAL库开发问题 1can初始化进入error handle2发送邮箱不能按照填写顺序发送3 GD32修改代码被stm32cudemx覆盖问题 1can初始化进入error handle HAL库的HAL_CAN_Init中,hcan->Instance->MSR寄存器无法清零,STM32先清零,再退出睡眠模…...

PE文件分析
用PEView及Python程序分析压缩包中的lab01.exe和lab01.dll两个PE文件,关键步骤给出截图: 分析两个文件的文件头中的各部分; 2.找出时间戳,计算出两个文件的生成时间; 注:时间戳为从1970年1月1日0时0分0秒到…...
synchronized 学习
学习源: https://www.bilibili.com/video/BV1aJ411V763?spm_id_from333.788.videopod.episodes&vd_source32e1c41a9370911ab06d12fbc36c4ebc 1.应用场景 不超卖,也要考虑性能问题(场景) 2.常见面试问题: sync出…...

大话软工笔记—需求分析概述
需求分析,就是要对需求调研收集到的资料信息逐个地进行拆分、研究,从大量的不确定“需求”中确定出哪些需求最终要转换为确定的“功能需求”。 需求分析的作用非常重要,后续设计的依据主要来自于需求分析的成果,包括: 项目的目的…...

Redis相关知识总结(缓存雪崩,缓存穿透,缓存击穿,Redis实现分布式锁,如何保持数据库和缓存一致)
文章目录 1.什么是Redis?2.为什么要使用redis作为mysql的缓存?3.什么是缓存雪崩、缓存穿透、缓存击穿?3.1缓存雪崩3.1.1 大量缓存同时过期3.1.2 Redis宕机 3.2 缓存击穿3.3 缓存穿透3.4 总结 4. 数据库和缓存如何保持一致性5. Redis实现分布式…...

基于当前项目通过npm包形式暴露公共组件
1.package.sjon文件配置 其中xh-flowable就是暴露出去的npm包名 2.创建tpyes文件夹,并新增内容 3.创建package文件夹...
linux 错误码总结
1,错误码的概念与作用 在Linux系统中,错误码是系统调用或库函数在执行失败时返回的特定数值,用于指示具体的错误类型。这些错误码通过全局变量errno来存储和传递,errno由操作系统维护,保存最近一次发生的错误信息。值得注意的是,errno的值在每次系统调用或函数调用失败时…...

Java-41 深入浅出 Spring - 声明式事务的支持 事务配置 XML模式 XML+注解模式
点一下关注吧!!!非常感谢!!持续更新!!! 🚀 AI篇持续更新中!(长期更新) 目前2025年06月05日更新到: AI炼丹日志-28 - Aud…...

CMake 从 GitHub 下载第三方库并使用
有时我们希望直接使用 GitHub 上的开源库,而不想手动下载、编译和安装。 可以利用 CMake 提供的 FetchContent 模块来实现自动下载、构建和链接第三方库。 FetchContent 命令官方文档✅ 示例代码 我们将以 fmt 这个流行的格式化库为例,演示如何: 使用 FetchContent 从 GitH…...

什么是Ansible Jinja2
理解 Ansible Jinja2 模板 Ansible 是一款功能强大的开源自动化工具,可让您无缝地管理和配置系统。Ansible 的一大亮点是它使用 Jinja2 模板,允许您根据变量数据动态生成文件、配置设置和脚本。本文将向您介绍 Ansible 中的 Jinja2 模板,并通…...
Git常用命令完全指南:从入门到精通
Git常用命令完全指南:从入门到精通 一、基础配置命令 1. 用户信息配置 # 设置全局用户名 git config --global user.name "你的名字"# 设置全局邮箱 git config --global user.email "你的邮箱example.com"# 查看所有配置 git config --list…...

Ubuntu Cursor升级成v1.0
0. 当前版本低 使用当前 Cursor v0.50时 GitHub Copilot Chat 打不开,快捷键也不好用,当看到 Cursor 升级后,还是蛮高兴的 1. 下载 Cursor 下载地址:https://www.cursor.com/cn/downloads 点击下载 Linux (x64) ,…...