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

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

640.png

空值合并运算符 '??'


空值合并运算符 ?? 提供了一种简短的语法,用来获取列表中第一个“已定义”的变量(译注:即值不是 nullundefined 的变量)。


a ?? b 的结果是:

  • a,如果 a 不是 nullundefined
  • b,其他情况。

所以,x = a ?? b 是下面这个表达式的简写:


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


下面是一个更长一点的例子。

假设,我们有一个用户,变量 firstNamelastNamenickName 分别对应用户的名字、姓氏和昵称。如果用户决定不输入任何值,那么这些变量都可能是未定义的。

我们想要显示用户的名称:显示这三个变量中的一个,如果都没有设置值,则显示 "Anonymous"。


让我们使用 ?? 运算符选择第一个已定义的变量:


let firstName = null;
let lastName = null;
let nickName = "Supercoder";
// 显示第一个不是 null/undefined 的值
alert(firstName ?? lastName ?? nickName ?? "Anonymous"); // Supercoder


与 || 比较


或运算符 || 可以与 ?? 运算符以同样的方式使用。正如 上一章[1] 所讲的,我们可以用 || 替换上面示例中的 ??,也可以获得相同的结果。


重要的区别是:


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


当我们想将 null/undefined0 区别对待时,这个区别至关重要。


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


height = height ?? 100;


如果 height 未定义,则将其赋值为 100

让我们将其与 || 进行比较:


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


在这个例子中,height || 100 将值为 0height 视为未设置的(unset),与 nullundefined 以及任何其他假(falsy)值同等对待。因此得到的结果是 100

height ?? 100 仅当 height 确实是 nullundefined 时才返回 100。因此,alert 按原样显示了 height0

哪种行为更好取决于特定的使用场景。当高度 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 x = 1 && 2 ?? 3; // Syntax error


这个限制无疑是值得商榷的,但是它被添加到语言规范中是为了避免编程错误,因为人们开始使用 ?? 替代 ||

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


let x = (1 && 2) ?? 3; // 起作用
alert(x); // 2


总结


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


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


  • ?? 运算符的优先级非常低,只略高于 ?=
  • 如果没有明确添加括号,不能将其与 ||&& 一起使用


目录
相关文章
|
2月前
ES6扩展运算符和剩余参数运算符
该文章讲解了ES6中扩展运算符和剩余参数运算符的用法,包括展开数组、浅拷贝、将伪数组转换为真数组,以及如何将函数调用时的参数收集到一个数组中。
32 3
|
3月前
|
JavaScript 前端开发
利用ES6中的...扩展运算符来合并数组
本文介绍了使用ES6的新特性简化JavaScript编程的多种方法。其中包括使用扩展运算符`...`优雅地合并数组,解构赋值快速提取数组和对象值,`for...of`循环及箭头函数提升数组遍历效率,`find`方法简化数组搜索,`+`号与短路运算符优化数据类型转换及默认值设置,以及模板字符串和扩展运算符增强的对象合并技巧,这些方法让代码更加简洁高效。
26 0
|
3月前
|
存储 API
ES6新增语法 扩展运算符
ES6新增语法 扩展运算符
26 0
|
5月前
|
C# 数据库
C#中的空合并运算符与空合并赋值运算符:简化空值处理
C#中的空合并运算符与空合并赋值运算符:简化空值处理
|
6月前
|
SQL 前端开发 关系型数据库
MYSQL基础知识之【LIKE子句的使用 ,NULL值的处理,空值的处理】
MYSQL基础知识之【LIKE子句的使用 ,NULL值的处理,空值的处理】
130 0
es6扩展运算符、concat方法合并多个数组
es6扩展运算符、concat方法合并多个数组
50 0
|
SQL JSON 数据格式
ES中如何实现空值和非空值的查询
ES中如何实现空值和非空值的查询
4716 0
|
JavaScript 前端开发
空值合并运算符和可选链
空值合并运算符和可选链
|
JavaScript
es6数据类型Symbol以及es6操作数组常用的方法
es6数据类型Symbol以及es6操作数组常用的方法
87 0
es6:空值合并运算符
es6:空值合并运算符
115 0
es6:空值合并运算符