Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[zh-cn]: sync with "Optional chaining" #24193

Merged
merged 4 commits into from
Oct 30, 2024
Merged
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
@@ -1,13 +1,13 @@
---
title: 可选链运算符(?.)
slug: Web/JavaScript/Reference/Operators/Optional_chaining
l10n:
sourceCommit: 8cb0caef8175e1772f13ef7bc761f9616e2c5a4b
---

{{JSSidebar("Operators")}}
{{jsSidebar("Operators")}}

**可选链运算符**(**`?.`**)允许读取位于连接对象链深处的属性的值,而不必明确验证链中的每个引用是否有效。`?.` 运算符的功能类似于 `.` 链式运算符,不同之处在于,在引用为空 ([nullish](/zh-CN/docs/Glossary/nullish) ) ({{JSxRef("null")}} 或者 {{JSxRef("undefined")}}) 的情况下不会引起错误,该表达式短路返回值是 `undefined`。与函数调用一起使用时,如果给定的函数不存在,则返回 `undefined`。

当尝试访问可能不存在的对象属性时,可选链运算符将会使表达式更短、更简明。在探索一个对象的内容时,如果不能确定哪些属性必定存在,可选链运算符也是很有帮助的。
**可选链运算符(`?.`)** 用于访问对象的属性或调用函数。如果使用此运算符访问的对象或调用的函数是 {{jsxref("undefined")}} 或 [`null`](/zh-CN/docs/Web/JavaScript/Reference/Operators/null),则表达式会短路并计算为 {{jsxref("undefined")}},而不是抛出错误。

{{EmbedInteractiveExample("pages/js/expressions-optionalchainingoperator.html", "taller")}}

Expand All @@ -21,151 +21,227 @@ obj.func?.(args)

## 描述

通过连接的对象的引用或函数可能是 `undefined` 或 `null` 时,可选链运算符提供了一种方法来简化被连接对象的值访问。
`?.` 运算符与 `.` 链式运算符相似,不同之处在于,如果引用是[空值](/zh-CN/docs/Glossary/Nullish)([`null`](/zh-CN/docs/Web/JavaScript/Reference/Operators/null) 或 {{jsxref("undefined")}}),它不会导致错误,而是使表达式短路并返回 `undefined`。当用于函数调用时,如果给定函数不存在,它也会返回 `undefined`。

当访问链式属性时,如果存在引用可能缺失的情况,这将使表达式更简洁、更简短。在探索对象内容且无法确定哪些属性是必需的时,它也会很有帮助。

比如,思考一个存在嵌套结构的对象 `obj`。不使用可选链的话,查找一个深度嵌套的子属性时,需要验证之间的引用,例如:
例如,考虑一个具有嵌套结构的对象 `obj`。如果没有可选链,查找深层嵌套的子属性需要验证中间的引用,例如:

```js
let nestedProp = obj.first && obj.first.second;
const nestedProp = obj.first && obj.first.second;
```

为了避免报错,在访问`obj.first.second`之前,要保证 `obj.first` 的值既不是 `null`,也不是 `undefined`。如果只是直接访问 `obj.first.second`,而不对 `obj.first` 进行校验,则有可能抛出错误。
在访问 `obj.first.second` 之前,要保证 `obj.first` 的值不是 `null`(也不是 `undefined`)。这样做可以避免在不对 `obj.first` 进行校验情况下直接访问 `obj.first.second` 而可能引发的错误。

这是 JavaScript 中的一个惯用模式,但当链很长时,它会变得冗长,且不安全。例如,如果 `obj.first` 是一个非 `null` 或 `undefined` 的 {{Glossary("Falsy", "假值")}},比如 `0`,它仍然会短路并使 `nestedProp` 变为 `0`,这可能是不可取的。
fuchunhui marked this conversation as resolved.
Show resolved Hide resolved

有了可选链运算符(`?.`),在访问 `obj.first.second` 之前,不再需要明确地校验 `obj.first` 的状态,再并用短路计算获取最终结果
然而,使用可选链运算符(`?.`),在访问 `obj.first.second` 之前,不再需要基于 `obj.first` 的状态进行明确的测试和短路操作了

```js
let nestedProp = obj.first?.second;
const nestedProp = obj.first?.second;
```

通过使用 `?.` 运算符取代 `.` 运算符,JavaScript 会在尝试访问 `obj.first.second` 之前,先隐式地检查并确定 `obj.first` 既不是 `null` 也不是 `undefined`。如果`obj.first` 是 `null` 或者 `undefined`,表达式将会短路计算直接返回 `undefined`。
fuchunhui marked this conversation as resolved.
Show resolved Hide resolved

这等价于以下表达式,但实际上没有创建临时变量:

```js
let temp = obj.first;
let nestedProp = temp === null || temp === undefined ? undefined : temp.second;
const temp = obj.first;
const nestedProp =
temp === null || temp === undefined ? undefined : temp.second;
```

### 可选链与函数调用
可选链运算符不能用于未声明的根对象,但可以用于值为 `undefined` 的根对象。

```js example-bad
undeclaredVar?.prop; // 引用错误:未定义的变量 undeclaredVar
fuchunhui marked this conversation as resolved.
Show resolved Hide resolved
```

当尝试调用一个可能不存在的方法时也可以使用可选链。这将是很有帮助的,比如,当使用一个 API 的方法可能不可用时,要么因为实现的版本问题要么因为当前用户的设备不支持该功能。
### 函数调用中的可选链

函数调用时如果被调用的方法不存在,使用可选链可以使表达式自动返回`undefined`而不是抛出一个异常。
当尝试调用一个可能不存在的方法时也可以使用可选链。例如,当使用某个 API 时,如果由于实现版本过旧或用户设备不具备某项功能而导致某个方法不可用时,可选链就很有用。

函数调用时如果被调用的方法不存在,使用可选链可以使表达式自动返回 `undefined` 而不是抛出一个异常。

```js
let result = someInterface.customMethod?.();
const result = someInterface.customMethod?.();
```

> [!NOTE]
> 如果存在一个属性名且不是函数,使用 `?.` 仍然会产生一个 {{JSxRef("TypeError")}} 异常 (`x.y is not a function`).
然而,如果存在一个具有这样名称的属性且不是函数,使用 `?.` 仍然会引发一个 {{jsxref("TypeError")}} 异常,即“`someInterface.customMethod` 不是一个函数”。
fuchunhui marked this conversation as resolved.
Show resolved Hide resolved

> [!NOTE]
> 如果 `someInterface` 自身是 `null` 或者 `undefined` ,异常 {{JSxRef("TypeError")}} 仍会被抛出 `someInterface is null` 如果你希望允许 `someInterface` 也为 `null` 或者 `undefined` ,那么你需要像这样写 `someInterface?.customMethod?.()`
> 如果 `someInterface` 自身是 `null` 或者 `undefined`,异常 {{JSxRef("TypeError")}} 仍会被抛出`someInterface is null`)。如果你希望允许 `someInterface` 自身也为 `null` 或者 `undefined`,你需要在这个位置使用 `?.`:`someInterface?.customMethod?.()`
fuchunhui marked this conversation as resolved.
Show resolved Hide resolved

#### 处理可选的回调函数或者事件处理器
`eval?.()` 是进入[_间接求值_](/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/eval#direct_and_indirect_eval)模式的最短方式。

如果使用[解构赋值](/zh-CN/docs/Web/JavaScript/Reference/Operators/Destructuring_assignment)来解构的一个对象的回调函数或 fetch 方法,你可能得到不能当做函数直接调用的不存在的值,除非你已经校验了他们的存在性。使用`?.`的你可以忽略这些额外的校验:
### 表达式中的可选链

你也可以将可选链操作符与[方括号表示法](/zh-CN/docs/Web/JavaScript/Reference/Operators/Property_accessors#方括号表示法)结合使用,它允许将表达式作为属性名传递:
fuchunhui marked this conversation as resolved.
Show resolved Hide resolved

```js
// ES2019 的写法
function doSomething(onContent, onError) {
try {
// ... do something with the data
} catch (err) {
if (onError) {
// 校验 onError 是否真的存在
onError(err.message);
}
}
}
const nestedProp = obj?.["prop" + "Name"];
```

这对于数组尤其有用,因为数组索引必须使用方括号来使用。

```js
// 使用可选链进行函数调用
function doSomething(onContent, onError) {
try {
// ... do something with the data
} catch (err) {
onError?.(err.message); // 如果 onError 是 undefined 也不会有异常
}
function printMagicIndex(arr) {
console.log(arr?.[42]);
}

printMagicIndex([0, 1, 2, 3, 4, 5]); // undefined
printMagicIndex(); // undefined;如果未使用 ?. 操作符,这将抛出一个错误:“无法读取未定义的属性(读取“42”时)”
fuchunhui marked this conversation as resolved.
Show resolved Hide resolved
```

### 无效的可选链

尝试为可选链表达式的结果赋值是无效的:

```js-nolint example-bad
const object = {};
object?.property = 1; // 语法错误:赋值中的左侧无效
fuchunhui marked this conversation as resolved.
Show resolved Hide resolved
```

[模板字符串标签](/zh-CN/docs/Web/JavaScript/Reference/Template_literals#带标签的模板)不能是可选链(参见[语法错误:带标签的模板不能与可选链一起使用](/zh-CN/docs/Web/JavaScript/Reference/Errors/Bad_optional_template)):
fuchunhui marked this conversation as resolved.
Show resolved Hide resolved

```js-nolint example-bad
String?.raw`Hello, world!`;
String.raw?.`Hello, world!`; // 语法错误:可选链上的标签模板无效
```

{{jsxref("Operators/new", "new")}} 表达式的构造函数不能是可选链(参见[语法错误:new 关键字不能与可选链一起使用](/zh-CN/docs/Web/JavaScript/Reference/Errors/Bad_new_optional)):
fuchunhui marked this conversation as resolved.
Show resolved Hide resolved

```js-nolint example-bad
new Intl?.DateTimeFormat(); // 语法错误:new 表达式的可选链无效
fuchunhui marked this conversation as resolved.
Show resolved Hide resolved
new Map?.();
```

### 可选链和表达式
### 短路

当使用[方括号与属性名](/zh-CN/docs/Web/JavaScript/Reference/Operators/Property_accessors#方括号表示法)的形式来访问属性时,你也可以使用可选链运算符
在使用可选链时,如果左操作数是 `null` 或 `undefined`,则表达式将不会被求值。例如

```js
let nestedProp = obj?.["prop" + "Name"];
const potentiallyNullObj = null;
let x = 0;
const prop = potentiallyNullObj?.[x++];

console.log(x); // x 未被递增,因此为 0
```

后续的属性访问也不会被求值。

```js
const potentiallyNullObj = null;
const prop = potentiallyNullObj?.a.b;
// 这不会抛出错误,因为求值已经在第一个可选链处停止了
```

这等价于:

```js
const potentiallyNullObj = null;
const prop =
potentiallyNullObj === null || potentiallyNullObj === undefined
? undefined
: potentiallyNullObj.a.b;
```

### 可选链不能用于赋值
然而,这种短路行为只会在一个连续的属性访问“链”中发生。如果你将链中的某一部分进行[分组](/zh-CN/docs/Web/JavaScript/Reference/Operators/Grouping),那么后续的属性访问仍然会被求值。

```plain
let object = {};
object?.property = 1; // Uncaught SyntaxError: Invalid left-hand side in assignment
```js
const potentiallyNullObj = null;
const prop = (potentiallyNullObj?.a).b;
// 类型错误:不能读取未定义的属性(读取“b”)
fuchunhui marked this conversation as resolved.
Show resolved Hide resolved
```

### 可选链访问数组元素
这等价于:

```plain
let arrayItem = arr?.[42];
```js
const potentiallyNullObj = null;
const temp = potentiallyNullObj?.a;
const prop = temp.b;
```

除了没有创建 `temp` 变量。

## 示例

### 基本例子
fuchunhui marked this conversation as resolved.
Show resolved Hide resolved

如下的例子在一个不含 `bar` 成员的 Map 中查找 `bar` 成员的 `name` 属性,因此结果是 `undefined`。
此示例在一个不含 `bar` 成员的 Map 中查找 `bar` 成员的 `name` 属性,因此结果是 `undefined`。

```js
let myMap = new Map();
const myMap = new Map();
myMap.set("foo", { name: "baz", desc: "inga" });

let nameBar = myMap.get("bar")?.name;
const nameBar = myMap.get("bar")?.name;
```

### 短路计算
### 处理可选的回调函数或者事件处理器
fuchunhui marked this conversation as resolved.
Show resolved Hide resolved

当在表达式中使用可选链时,如果左操作数是 `null` 或 `undefined`,表达式将不会被计算,例如
如果使用[解构赋值](/zh-CN/docs/Web/JavaScript/Reference/Operators/Destructuring_assignment)来解构的一个对象的回调函数或 fetch 方法,你可能得到不能当做函数直接调用的不存在的值,除非你已经校验了他们的存在性。使用`?.`的你可以忽略这些额外的校验
fuchunhui marked this conversation as resolved.
Show resolved Hide resolved

```js
let potentiallyNullObj = null;
let x = 0;
let prop = potentiallyNullObj?.[x++];
// 不使用可选链的写法
function doSomething(onContent, onError) {
try {
// 用数据做些事情
} catch (err) {
// 校验 onError 是否真的存在
if (onError) {
onError(err.message);
}
}
}
```

console.log(x); // x 将不会被递增,依旧输出 0
```js
// 使用可选链进行函数调用
function doSomething(onContent, onError) {
try {
// 用数据做些事情
} catch (err) {
onError?.(err.message); // 如果 onError 是 undefined 也不会有异常
}
}
```

### 连用可选链运算符

可以连续使用可选链读取多层嵌套结构
在嵌套结构中,可以多次使用可选链

```js
let customer = {
const customer = {
name: "Carl",
details: {
age: 82,
location: "Paradise Falls", // details 的 address 属性未有定义
},
};
let customerCity = customer.details?.address?.city;
const customerCity = customer.details?.address?.city;

// 可选链也可以和函数调用一起使用
let duration = vacations.trip?.getTime?.();
// 可选链也可以和函数调用一起使用
const customerName = customer.name?.getName?.(); // 方法不存在,customerName 未定义
```

### 使用空值合并运算符

{{JSxRef("Operators/Nullish_Coalescing_Operator", "空值合并运算符")}}可以在使用可选链时设置一个默认值
当未找到任何值时,可以在可选链之后使用[空值合并运算符](/zh-CN/docs/Web/JavaScript/Reference/Operators/Nullish_coalescing)来构建一个默认值

```js
let customer = {
function printCustomerCity(customer) {
const customerCity = customer?.city ?? "Unknown city";
fuchunhui marked this conversation as resolved.
Show resolved Hide resolved
console.log(customerCity);
}

printCustomerCity({
name: "Nathan",
city: "Paris",
}); // "Paris"
printCustomerCity({
name: "Carl",
details: { age: 82 },
};
let customerCity = customer?.city ?? "暗之城";
console.log(customerCity); // “暗之城”
}); // "Unknown city"
fuchunhui marked this conversation as resolved.
Show resolved Hide resolved
```

## 规范
Expand All @@ -178,4 +254,4 @@ console.log(customerCity); // “暗之城”

## 参见

- [空值合并运算符(??)](/zh-CN/docs/Web/JavaScript/Reference/Operators/Nullish_coalescing_operator)
- [空值合并运算符(??)](/zh-CN/docs/Web/JavaScript/Reference/Operators/Nullish_coalescing)