空值合并运算符 ??

  • 2021.06.02

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

TIP

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

我们来看看执行的结果,a ?? b 的结果是:

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

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

换做以前的写法也就是:

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

使用场景

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

比如,由于 user 没有定义赋值一个默认值。

(function() {
  let user;

  return user ?? "Jack"; // Jack
})();

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

(function() {
  let user = "";

  return user ?? "Jack"; // 空字符串
})();

||运算符 比较

某些场景下或运算符 || 可以以与 ?? 运算符相同的方式使用。

例如上面的代码:

(function() {
  let user;

  return user || "Jack"; // Jack
})();

但是它们之间重要的区别是:

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

还是上面的那个例子,假如我们改成或运算符的写法,就会得到不一样的结果:

(function() {
  let user = "";

  return user || "Jack"; // Jack
})();

TIP

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

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

优先级

?? 运算符的优先级相当低:在 MDN table 中为 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;
    
  • ?? 运算符的优先级非常低,仅略高于 ?=,因此在表达式中使用它时请考虑添加括号。

  • 如果没有明确添加括号,不能将其与 ||&& 一起使用。

上次更新时间: 2021-06-07 09:34:00