《你不知道的Javascript系列》——不可变性immutable

2023-10-29 13:20

本文主要是介绍《你不知道的Javascript系列》——不可变性immutable,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

基础

JavaScript中存在以下数据类型:

  1. 原生(基本)类型 —— Boolean, Number, String
  2. 非原始(引用)类型或对象 —— Object, Array, Function
  3. 特殊 —— Null, Undefined

原生数据类型默认是不可变的,对于常量声明的原生类型,他们值是不可变的,若用变量声明,虽然可以用=改变其值,实际上是重新赋值,仍没有改变内存地址所存的值,以下面代码为例:

let var1 = 'apple' //'apple' is stored in memory location A
var1 = 'orange' //'orange' is stored in memory location Bconst var2 = 'apple'
var2 = 'orange' // ERROR: Re-assignment not allowed for constants

在上述例子中,如果我们修改 var1 这个 string,JavaScript 将会在内存中的另一个位置创造另一个 string,而 var1 将会指向这个新的内存位置,这被称为 重新赋值

因此重新赋值并不代表其为可变的,所有原生类型均为不可变的,这也就是为什么string类型的方法均会返回新的string,而不是改变原来的string。

对于非原始类型来说,是可变的。比如下面这段代码,改变了profile1指向的内存中的数据的属性:

const profile1 = {'username':'peter'}
profile1.username = 'tom'
console.log(profile1) //{'username':'tom'}--------------------------------------------------
const sampleprofile = {'username':'name', 'pw': '123'}
const profile2 = sampleprofileprofile2.username = 'harry'console.log(profile2) // {'username':'harry', 'pw': '123'}
console.log(sampleprofile) // {'username':'harry', 'pw': '123'}

并且JavaScript是引用传递的(通过=将sampleprofile的引用传递给了profile2,此时二者指向的是内存中的同一数据),因此修改一个另一个也会修改。

理解

对于mutable(可变)和immutable(不可变)来说,本质区别在于变量指向的内存地址数据能否被修改。

  • 以immutable类型来说,用=重新赋值后,内存中的实际数据是不可修改的,只是重新在新的内存地址创建新的值,然后改变指针的指向
  • 以mutable类型来说,内存中的实际数据是可以被修改的,因此重新赋值后,二者所指的数据都发生了变化。

为了防止修改mutable类型的数据,有两种解决方案:

  • 通过冻结对象来防止修改
  • 使用浅拷贝和深拷贝

let 与 const

变量与常量,对常量来说,指的是指向的数据的内存地址指针不可变,而对变量来说,指针地址是可以变化的。

防止修改对象

freeze

使用 Object.freeze() 。它的作用是,防止对象已有的属性被改变。任何改变的尝试都会静默失败,意味着它不会成功,也不会有任何警告。不过是一种浅冻结,即它对于深层嵌套的对象将不会有用。

const sampleprofile = {'username':'name','pw': '123','particulars':{'firstname':'name', 'lastname':'name'}
}Object.freeze(sampleprofile)sampleprofile.username = 'another name' // no effectsampleprofile.particulars.firstname = 'changedName' // changes

拷贝

浅拷贝

扩展操作符...

所有对象的属性将被合并在一起,但对于冲突的属性,后展开的对象有更高的优先级。

const profile1 = {'username':'name', 'pw': '123', 'age': 16}
const profile2 = {'username':'tom', 'pw': '1234'}
const profile1Copy = {...profile1}
const resultProfile = {...profile1, ...profile2}console.log(profile1Copy) // {'username':'name', 'pw': '123', 'age': 16}
console.log(resultProfile) // {'username':'tom', 'pw': '1234', 'age': 16}

Object.assign()

类似于扩展操作符

Array.slice()

浅克隆数组的一个便捷方法

const firstSet = [1, 2, 3];
const firstSetCopy = firstSet.slice()console.log(firstSetCopy) // [1, 2, 3]//note that they are not the same objects
console.log(firstSet===firstSetCopy) // false

深拷贝

使用 JSON.stringify() 和 JSON.parse()

JSON.parse(JSON.stringfy({}))

使用 lodash.deepclone()

这篇关于《你不知道的Javascript系列》——不可变性immutable的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Java实现Excel与HTML互转

《Java实现Excel与HTML互转》Excel是一种电子表格格式,而HTM则是一种用于创建网页的标记语言,虽然两者在用途上存在差异,但有时我们需要将数据从一种格式转换为另一种格式,下面我们就来看看... Excel是一种电子表格格式,广泛用于数据处理和分析,而HTM则是一种用于创建网页的标记语言。虽然两

java图像识别工具类(ImageRecognitionUtils)使用实例详解

《java图像识别工具类(ImageRecognitionUtils)使用实例详解》:本文主要介绍如何在Java中使用OpenCV进行图像识别,包括图像加载、预处理、分类、人脸检测和特征提取等步骤... 目录前言1. 图像识别的背景与作用2. 设计目标3. 项目依赖4. 设计与实现 ImageRecogni

Java中Springboot集成Kafka实现消息发送和接收功能

《Java中Springboot集成Kafka实现消息发送和接收功能》Kafka是一个高吞吐量的分布式发布-订阅消息系统,主要用于处理大规模数据流,它由生产者、消费者、主题、分区和代理等组件构成,Ka... 目录一、Kafka 简介二、Kafka 功能三、POM依赖四、配置文件五、生产者六、消费者一、Kaf

Java访问修饰符public、private、protected及默认访问权限详解

《Java访问修饰符public、private、protected及默认访问权限详解》:本文主要介绍Java访问修饰符public、private、protected及默认访问权限的相关资料,每... 目录前言1. public 访问修饰符特点:示例:适用场景:2. private 访问修饰符特点:示例:

详解Java如何向http/https接口发出请求

《详解Java如何向http/https接口发出请求》这篇文章主要为大家详细介绍了Java如何实现向http/https接口发出请求,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 用Java发送web请求所用到的包都在java.net下,在具体使用时可以用如下代码,你可以把它封装成一

SpringBoot使用Apache Tika检测敏感信息

《SpringBoot使用ApacheTika检测敏感信息》ApacheTika是一个功能强大的内容分析工具,它能够从多种文件格式中提取文本、元数据以及其他结构化信息,下面我们来看看如何使用Ap... 目录Tika 主要特性1. 多格式支持2. 自动文件类型检测3. 文本和元数据提取4. 支持 OCR(光学

Java内存泄漏问题的排查、优化与最佳实践

《Java内存泄漏问题的排查、优化与最佳实践》在Java开发中,内存泄漏是一个常见且令人头疼的问题,内存泄漏指的是程序在运行过程中,已经不再使用的对象没有被及时释放,从而导致内存占用不断增加,最终... 目录引言1. 什么是内存泄漏?常见的内存泄漏情况2. 如何排查 Java 中的内存泄漏?2.1 使用 J

JAVA系统中Spring Boot应用程序的配置文件application.yml使用详解

《JAVA系统中SpringBoot应用程序的配置文件application.yml使用详解》:本文主要介绍JAVA系统中SpringBoot应用程序的配置文件application.yml的... 目录文件路径文件内容解释1. Server 配置2. Spring 配置3. Logging 配置4. Ma

Java 字符数组转字符串的常用方法

《Java字符数组转字符串的常用方法》文章总结了在Java中将字符数组转换为字符串的几种常用方法,包括使用String构造函数、String.valueOf()方法、StringBuilder以及A... 目录1. 使用String构造函数1.1 基本转换方法1.2 注意事项2. 使用String.valu

java脚本使用不同版本jdk的说明介绍

《java脚本使用不同版本jdk的说明介绍》本文介绍了在Java中执行JavaScript脚本的几种方式,包括使用ScriptEngine、Nashorn和GraalVM,ScriptEngine适用... 目录Java脚本使用不同版本jdk的说明1.使用ScriptEngine执行javascript2.