一文彻底搞懂 JS 中的基础类型和引用类型

简介: 一文彻底搞懂 JS 中的基础类型和引用类型

在JavaScript中,可以按值和按引用传递

这两种方法的主要区别在于,在赋值原始值(primitive)时是传递值,而在赋值对象(objects)时是传递引用

让我们在这篇文章中更详细地讨论值和引用。

primitive && objects

JavaScript提供了两类数据类型: 基本类型(primitive)和引用类型(objects)。

基本类型包括数字、布尔值、字符串、符号和特殊值(null和undefined)。例如:

// Primitives
const number = 10;
const bool = false;
const str = 'Hello!';
const missingObject = null;
const nothing = undefined;

第二类是对象。特别是普通对象、数组、函数等等都是对象

// Objects
const plainObject = {
  prop: 'Value'
};
const array = [1, 5, 6];
const functionObject = (n1, n2) => {
  return n1 + n2;
};

换句话说,任何不是原始值的东西都是对象。

值传递

按值传递的简单规则是,JavaScript中的所有原始值都是按值传递的。就这么简单。

通过值传递意味着每次给变量赋值时,都会创建该值的一个副本

假设你有两个变量a和b:

let a = 1;
let b = a;
b = b + 2;
console.log(a); // 1
console.log(b); // 3

第一个语句让a = 1定义一个初始化为数字1的变量a。

第二个语句让b = a定义另一个变量b,并使用一个变量的值初始化它——这是通过value传递的。简单点说,把数字1的一个副本赋给b

之后,b = b + 2增加2,变成3。变量b发生变化,这个变化不影响变量a的值

引用传递

然而,引用传递的表现方式不同。

当创建一个对象时,你会得到一个对该对象的引用如果两个变量拥有相同的引用,那么改变对象会反映在两个变量中

让我们检查下面的代码示例:

let x = [1];
let y = x;
y.push(2);
console.log(x); // [1, 2]
console.log(y); // [1, 2]

第一个语句让x =[1]创建一个数组,定义一个变量x,并使用对创建的数组的引用初始化该变量。

然后让y = x定义一个变量y,并使用存储在x变量中的引用初始化y。这是一个引用传递。

y.push(2)通过入项2来改变数组。因为xy变量引用相同的数组,所以这一变化反映在两个变量中。

注意: 为简单起见,我说变量包含对对象的引用。但是严格地说,JavaScript中的变量保存的值是对对象的引用

使用比较运算符

在比较对象时,理解值和引用之间的差异很重要。

当使用严格比较操作符===时,如果两个变量的值相同,那么它们就是相等的。下面所有的比较都是相等的:

const one = 1;
const oneCopy = 1;
console.log(one === oneCopy); // true
console.log(one === 1);       // true
console.log(one === one);     // true

oneoneCopy的值相同。当两个操作数的长度都为1时,操作符===的计算结果为true,而不管值是从哪里取的。

但是比较操作符===比较引用时的工作方式不同只有当它们引用完全相同的对象时,两个引用才相等

例如,ar1和ar2引用不同的数组:

const ar1 = [1];
const ar2 = [1];
console.log(ar1 === ar2); // false
console.log(ar1 === [1]);  // false
const ar11 = ar1;
console.log(ar1 === ar11); // true
console.log(ar1 === ar1);  // true

ar1和ar2引用相同结构的数组,但是ar1 === ar2计算为false,因为ar1和ar2引用不同的数组对象

比较操作符仅在比较指向同一个对象的引用时返回true: ar1 === ar11ar1 === ar1

总结

在JavaScript中,基本类型是作为值传递的:这意味着每次赋值时,都会创建一个值的副本

另一方面,对象(包括普通对象、数组、函数、类实例)是引用。如果您修改了该对象,那么所有引用该对象的变量都将看到更改

比较运算符区分比较值和引用。只有引用了完全相同的对象时,两个引用的变量才相等,但如果两个变量的值是相同的,那么它们的值就是相等的,而不管值来自哪个地方。



相关文章
|
1月前
|
JavaScript 前端开发 安全
使用TypeScript增强JavaScript应用的类型安全性
【5月更文挑战第23天】TypeScript是微软开发的JavaScript超集,引入静态类型检查和面向对象特性,提升代码可维护性和可靠性。它在编译阶段捕获类型错误,增强代码可读性,并通过接口、类、泛型和类型断言等工具确保类型安全。使用TypeScript能有效避免复杂项目中的调试难题,尤其适合大型项目。
|
1天前
|
存储 JavaScript 前端开发
JavaScript中的对象是数据结构,存储键值对,键为字符串,值可为任意类型,包括函数(作为方法)
【6月更文挑战第25天】JavaScript中的对象是数据结构,存储键值对,键为字符串,值可为任意类型,包括函数(作为方法)。
8 2
|
15天前
|
前端开发 JavaScript 安全
TypeScript作为一种静态类型的JavaScript超集,其强大的类型系统和面向对象编程特性为微前端架构的实现提供了有力的支持
【6月更文挑战第11天】微前端架构借助TypeScript提升开发效率和代码可靠性。 TypeScript提供类型安全,防止微前端间通信出错;智能提示和自动补全加速跨代码库开发;重构支持简化代码更新。通过定义公共接口确保一致性,用TypeScript编写微前端以保证质量。集成到构建流程确保顺利构建打包。在微前端场景中,TypeScript是强有力的语言选择。
29 2
|
1月前
|
存储 JavaScript 前端开发
JavaScript 中松散类型的理解
JavaScript 中松散类型的理解
15 1
|
1月前
|
JavaScript 前端开发
JS中运算符的算术、赋值、+、比较(不同类型之间比较)、逻辑
JS中运算符的算术、赋值、+、比较(不同类型之间比较)、逻辑
19 1
|
1月前
|
JavaScript 前端开发 Java
javascript是弱类型语言,一个函数参数可以接收不同类型的变量作为它的该参数
javascript是弱类型语言,一个函数参数可以接收不同类型的变量作为它的该参数
30 0
|
1月前
|
前端开发 JavaScript 索引
【Web 前端】JS的几种具体异常类型(报错)
【4月更文挑战第22天】【Web 前端】JS的几种具体异常类型(报错)
|
1月前
|
JavaScript 前端开发
前端 JS 经典:JS 基础类型和 typeof
前端 JS 经典:JS 基础类型和 typeof
22 0
|
1月前
|
Web App开发 前端开发 JavaScript
在 Chrome 开发者工具里配置哪些类型的 JavaScript 文件应该被调试器忽略
在 Chrome 开发者工具里配置哪些类型的 JavaScript 文件应该被调试器忽略
|
1月前
|
数据可视化 JavaScript 前端开发
Echarts是一个开源的JavaScript可视化库,用于创建各种类型的图表
Echarts是JavaScript的开源可视化库,Python通过Pyecharts库可调用它来绘制图表。示例展示了如何用Pyecharts创建柱状图:定义图表对象,设置标题和坐标轴,添加X轴、Y轴数据,最后渲染展示。Pyecharts还支持折线图、散点图、饼图等多种图表类型,更多详情可查阅官方文档。
56 0