ES2020 系列:空值合并运算符 '??'

简介: ES2020 系列:空值合并运算符 '??'

640.png

空值合并运算符 '??'


在本文中,我们将值既不是 null 也不是 undefined 的表达式称为“已定义的(defined)”。


空值合并运算符(nullish coalescing operator)的写法为两个问号 ??

a ?? b 的结果是:


  • 如果 a 是已定义的,则结果为 a
  • 如果 a 不是已定义的,则结果为 b


换句话说,如果第一个参数不是 null/undefined,则 ?? 返回第一个参数。否则,返回第二个参数。


空值合并运算符并不是什么全新的东西。它只是一种获得两者中的第一个“已定义的”值的不错的语法。


我们可以使用我们已知的运算符重写 result = a ?? b,像这样:


result = (a !== null && a !== undefined) ? a : b;


通常 ?? 的使用场景是,为可能是未定义的变量提供一个默认值。

例如,在这里,如果 user 是未定义的,我们则显示 Anonymous


let user;
alert(user ?? "Anonymous"); // Anonymous


当然,如果 user 的值为除 null/undefined 外的任意值,那么我们看到的将是它:


let user = "John";
alert(user ?? "Anonymous"); // John


我们还可以使用 ?? 序列从一系列的值中选择出第一个非 null/undefined 的值。


假设我们在变量 firstNamelastNamenickName 中存储着一个用户的数据。如果用户决定不输入值,则所有这些变量的值都可能是未定义的。


我们想使用这些变量之一显示用户名,如果这些变量的值都是未定义的,则显示 "Anonymous"。


让我们使用 ?? 运算符来实现这一需求:


let firstName = null;
let lastName = null;
let nickName = "Supercoder";
// 显示第一个已定义的值
alert(firstName ?? lastName ?? nickName ?? "Anonymous"); // Supercoder


与 || 比较


或运算符 || 可以以与 ?? 运算符相同的方式使用。像我们在 上一章[1] 所讲的那样。

例如,在上面的代码中,我们可以用 || 替换掉 ??,也可以获得相同的结果:


let firstName = null;
let lastName = null;
let nickName = "Supercoder";
// 显示第一个真值:
alert(firstName || lastName || nickName || "Anonymous"); // Supercoder


|| 运算符自 JavaScript 诞生就存在,因此开发者长期将其用于这种目的。

另一方面,空值合并运算符 ?? 是最近才被添加到 JavaScript 中的,它的出现是因为人们对 || 不太满意。


它们之间重要的区别是:


  • || 返回第一个 值。
  • ?? 返回第一个 已定义的 值。


换句话说,|| 无法区分 false0、空字符串 ""null/undefined。它们都一样 —— 假值(falsy values)。如果其中任何一个是 || 的第一个参数,那么我们将得到第二个参数作为结果。


不过在实际中,我们可能只想在变量的值为 null/undefined 时使用默认值。也就是说,当该值确实未知或未被设置时。


例如,考虑下面这种情况:


let height = 0;
alert(height || 100); // 100
alert(height ?? 100); // 0


  • height || 100 首先会检查 height 是否为一个假值,发现它确实是。
  • 所以,结果为第二个参数,100
  • height ?? 100 首先会检查 height 是否为 null/undefined,发现它不是。
  • 所以,结果为 height 的原始值,0


如果高度 0 为有效值,则不应将其替换为默认值,所以 ?? 能够得出正确的结果。


优先级


?? 运算符的优先级相当低:在 MDN table[2] 中为 5。因此,??=? 之前计算,但在大多数其他运算符(例如,+*)之后计算。


因此,如果我们需要在还有其他运算符的表达式中使用 ?? 进行取值,需要考虑加括号:


let height = null;
let width = null;
// 重要:使用括号
let area = (height ?? 100) * (width ?? 50);
alert(area); // 5000


否则,如果我们省略了括号,则由于 * 的优先级比 ?? 高,它会先执行,进而导致错误的结果。


// 没有括号
let area = height ?? 100 * width ?? 50;
// ……与下面这行代码的计算方式相同(应该不是我们所期望的):
let area = height ?? (100 * width) ?? 50;


??&&|| 一起使用


出于安全原因,JavaScript 禁止将 ?? 运算符与 &&|| 运算符一起使用,除非使用括号明确指定了优先级。


下面的代码会触发一个语法错误:


let x = 1 && 2 ?? 3; // Syntax error


这个限制无疑是值得商榷的,但它被添加到语言规范中是为了避免人们从 || 切换到 ?? 时的编程错误。

可以明确地使用括号来解决这个问题:


let x = (1 && 2) ?? 3; // 正常工作了
alert(x); // 2


总结


  • 空值合并运算符 ?? 提供了一种从列表中选择第一个“已定义的”值的简便方式。
    它被用于为变量分配默认值:


// 当 height 的值为 null 或 undefined 时,将 height 的值设置为 100
height = height ?? 100;


  • ?? 运算符的优先级非常低,仅略高于 ?=,因此在表达式中使用它时请考虑添加括号。
  • 如果没有明确添加括号,不能将其与 ||&& 一起使用。
目录
相关文章
|
存储 算法 关系型数据库
深入理解InnoDB索引数据结构和算法
1. **索引定义**:索引是提升查询速度的有序数据结构,帮助数据库系统快速找到数据。 2. **索引类型**:包括普通索引、唯一索引、主键索引、空间索引和全文索引,每种有特定应用场景。 3. **数据结构**:InnoDB使用B+树作为索引结构,确保所有节点按顺序排列,降低查询时的磁盘I/O。 4. **B+树特性**:所有数据都在叶子节点,非叶子节点仅存储索引,提供高效范围查询。 5. **索引优势**:通过减少查找数据所需的磁盘I/O次数,显著提高查询性能。 **总结:**InnoDB索引通过B+树结构,优化了数据访问,使得查询速度快,尤其适合大数据量的场景。
863 0
深入理解InnoDB索引数据结构和算法
|
C# 容器
WPF技术之HorizontalAlignment和VerticalAlignment
HorizontalAlignment和VerticalAlignment是用来确定控件在其父容器中的水平和垂直位置的属性。
1656 0
WPF技术之HorizontalAlignment和VerticalAlignment
你的应用进入了中断状态,但无任何代码显示,因为所有线程之前都在执行外部代码
你的应用进入了中断状态,但无任何代码显示,因为所有线程之前都在执行外部代码
3119 0
你的应用进入了中断状态,但无任何代码显示,因为所有线程之前都在执行外部代码
|
5月前
|
人工智能 自然语言处理 数据可视化
DeepSeek+Coze:普通人也能轻松搭建AI智能体的完整指南优雅草卓伊凡
DeepSeek+Coze:普通人也能轻松搭建AI智能体的完整指南优雅草卓伊凡
1803 1
DeepSeek+Coze:普通人也能轻松搭建AI智能体的完整指南优雅草卓伊凡
|
11月前
|
关系型数据库 MySQL 数据库
mysql查看用户的过期时间
通过本文的介绍,希望您能够深入理解和掌握在MySQL中查看用户过期时间的方法,并在实际项目中灵活运用这些技术,提升数据库管理的安全性和效率。
758 3
|
弹性计算 监控 Serverless
函数计算操作报错合集之调用不成功,报错:Function instance health check failed on port 9000 in 120.7 seconds.该怎么办
在使用函数计算服务(如阿里云函数计算)时,用户可能会遇到多种错误场景。以下是一些常见的操作报错及其可能的原因和解决方法,包括但不限于:1. 函数部署失败、2. 函数执行超时、3. 资源不足错误、4. 权限与访问错误、5. 依赖问题、6. 网络配置错误、7. 触发器配置错误、8. 日志与监控问题。
331 0
|
小程序 开发者
微信开发者工具如何集成SVN、同时解决集成SVN无效果的问题
这篇文章介绍了如何在微信开发者工具中集成SVN版本控制系统,并提供了解决集成后无效果问题的方法。
微信开发者工具如何集成SVN、同时解决集成SVN无效果的问题
Visual Studio 快速分析 .NET Dump 文件
【11月更文挑战第10天】.NET Dump 文件是在 .NET 应用程序崩溃或出现问题时生成的,记录了应用程序的状态,包括内存对象、线程栈和模块信息。通过分析这些文件,开发人员可以定位和解决内存泄漏、死锁等问题。在 Visual Studio 中,可以通过调试工具、内存分析工具和符号加载等功能来详细分析 Dump 文件。此外,还可以使用第三方工具如 WinDbg 进行更深入的分析。
673 1
|
机器学习/深度学习 数据采集 Python
从零到一:手把手教你完成机器学习项目,从数据预处理到模型部署全攻略
【10月更文挑战第25天】本文通过一个预测房价的案例,详细介绍了从数据预处理到模型部署的完整机器学习项目流程。涵盖数据清洗、特征选择与工程、模型训练与调优、以及使用Flask进行模型部署的步骤,帮助读者掌握机器学习的最佳实践。
753 1
|
10月前
|
NoSQL 前端开发 Redis
单点登录云平台子系统集成方式
单点登录云平台子系统集成方式
377 0