From 3065c99257902d19461d30c00eedc0b9ff6246d8 Mon Sep 17 00:00:00 2001 From: allo Date: Thu, 6 Jan 2022 10:48:44 +0800 Subject: convert to markdown and sync with english version --- .../operators/operator_precedence/index.md | 141 ++++++++++----------- 1 file changed, 65 insertions(+), 76 deletions(-) (limited to 'files/zh-cn/web') diff --git a/files/zh-cn/web/javascript/reference/operators/operator_precedence/index.md b/files/zh-cn/web/javascript/reference/operators/operator_precedence/index.md index bb742caac7..47ad48e680 100644 --- a/files/zh-cn/web/javascript/reference/operators/operator_precedence/index.md +++ b/files/zh-cn/web/javascript/reference/operators/operator_precedence/index.md @@ -7,34 +7,37 @@ tags: - 优先级 translation_of: Web/JavaScript/Reference/Operators/Operator_Precedence --- -
{{jsSidebar("Operators")}}
+{{jsSidebar("Operators")}} -

运算符的优先级决定了表达式中运算执行的先后顺序。优先级高的运算符会作为优先级低的运算符的操作符。

+**运算符的优先级**决定了表达式中运算执行的先后顺序。优先级高的运算符会作为优先级低的运算符的操作符。 -
{{EmbedInteractiveExample("pages/js/expressions-operatorprecedence.html")}}
+{{EmbedInteractiveExample("pages/js/expressions-operatorprecedence.html")}} +## 优先级和结合性 -

优先级和结合性

+考虑由下面的表示法描述的表达式。其中,OP1 和 OP2 都是操作符的占位符。 -

考虑由下面的表示法描述的表达式。其中,OP1 和 OP2 都是操作符的占位符。

+```js +a OP1 b OP2 c +``` -
a OP1 b OP2 c
+如果 `OP1` 和 `OP2` 具有不同的优先级(见下表),则优先级最高的运算符先执行,不用考虑结合性。观察乘法如何具有比加法更高的优先级并首先执行,即使加法是首先写入代码的。 -

如果 OP1OP2 具有不同的优先级(见下表),则优先级最高的运算符先执行,不用考虑结合性。观察乘法如何具有比加法更高的优先级并首先执行,即使加法是首先写入代码的。

- -
console.log(3 + 10 * 2);   // 输出 23
+```js
+console.log(3 + 10 * 2);   // 输出 23
 console.log(3 + (10 * 2)); // 输出 23 因为这里的括号是多余的
 console.log((3 + 10) * 2); // 输出 26 因为括号改变了优先级
-
+``` -

左结合(左到右)相当于把左边的子表达式加上小括号 (a OP b) OP c,右结合(右到左)相当于 a OP (b OP c)。赋值运算符是右结合的,所以你可以这么写:

+左结合(左到右)相当于把左边的子表达式加上小括号 `(a OP b) OP c`,右结合(右到左)相当于 `a OP (b OP c)`。赋值运算符是右结合的,所以你可以这么写: -
a = b = 5; // 相当于 a = (b = 5);
-
+```js +a = b = 5; // 相当于 a = (b = 5); +``` -

预期结果是 a 和 b 的值都会成为 5。这是因为赋值运算符的返回结果就是赋值运算符右边的那个值,具体过程是:首先 b 被赋值为 5,然后 a 也被赋值为 b = 5 的返回值,也就是 5。

+预期结果是 `a` 和 `b` 的值都会成为 5。这是因为赋值运算符的返回结果就是赋值运算符右边的那个值,具体过程是:首先 `b` 被赋值为 5,然后 `a` 也被赋值为 `b = 5` 的返回值,也就是 5。 -

另一个例子是,只有幂运算符是右结合的,而其他算术运算符都是左结合的。有趣的是,无论结合性和优先级如何,求值顺序总是从左到右。

+另一个例子是,只有幂运算符是右结合的,而其他算术运算符都是左结合的。有趣的是,无论结合性和优先级如何,求值顺序总是从左到右。 @@ -81,7 +84,7 @@ Evaluating the right side
-

当有多个具有相同优先级的运算符时,结合性的差异就会发挥作用。仅使用一个或多个不同优先级的运算符时,结合性不会影响输出,如上面的例子所示。在下面的示例中,观察使用多个相同运算符时结合性会如何影响输出。

+当有多个具有相同优先级的运算符时,结合性的差异就会发挥作用。仅使用一个或多个不同优先级的运算符时,结合性不会影响输出,如上面的例子所示。在下面的示例中,观察使用多个相同运算符时结合性会如何影响输出。 @@ -150,43 +153,40 @@ Evaluating the right side
-

观察上面的代码片段,6 / 3 / 2 与 - (6 / 3) / 2 是相同的,因为除法是左结合的。而幂运算符是右结合的,所以 2 ** 3 ** 2 与 - 2 ** (3 ** 2) 是相同的。因此,(2 ** 3) ** 2 会更改执行顺序,并导致输出上表中的 64。

+观察上面的代码片段,`6 / 3 / 2` 与 `(6 / 3) / 2` 是相同的,因为除法是左结合的。而幂运算符是右结合的,所以 `2 ** 3 ** 2` 与 `2 ** (3 ** 2)` 是相同的。因此,`(2 ** 3) ** 2` 会更改执行顺序,并导致输出上表中的 64。 -

请记住,判断执行顺序的时候,优先级在结合性之前。所以,混合求除法和幂时,求幂会先于除法。例如, - 2 ** 3 / 3 ** 2 的结果是 0.8888888888888888,因为它相当于 - (2 ** 3) / (3 ** 2)

+请记住,判断执行顺序的时候,优先级在结合性之前。所以,混合求除法和幂时,求幂会先于除法。例如,`2 ** 3 / 3 ** 2` 的结果是 0.8888888888888888,因为它相当于 `(2 ** 3) / (3 ** 2)`。 -

分组和短路的注意事项

+### 分组和短路的注意事项 -

在下表中,分组(Grouping)具有最高优先级。然而,这并不意味着总是优先对分组符号 ( … ) 内的表达式进行求值,尤其是涉及短路时。

+在下表中,**分组(Grouping)** 具有最高优先级。然而,这并不意味着总是优先对分组符号 `( … )` 内的表达式进行求值,尤其是涉及短路时。 -

短路是条件求值的术语。例如,在表达式 - a && (b + c) 中,如果 a 为虚值({{Glossary("falsy")}}),那么即使 (b + c) 在圆括号中,也不会被求值。我们可以说逻辑或运算符(“OR”)是“短路的”。除了逻辑或运算符外,其他短路运算符还包括逻辑与(“AND”)、空值合并、可选链和条件(三元)运算符。下面有更多例子:

+短路是条件求值的术语。例如,在表达式 `a && (b + c)` 中,如果 `a` 为虚值({{Glossary("falsy")}}),那么即使 `(b + c)` 在圆括号中,也不会被求值。我们可以说逻辑或运算符(“OR”)是“短路的”。除了逻辑或运算符外,其他短路运算符还包括逻辑与(“AND”)、空值合并、可选链和条件(三元)运算符。下面有更多例子: -
a || (b * c);  // 首先对 `a` 求值,如果 `a` 为真值则直接返回 `a`
-a && (b < c);  // 首先对 `a` 求值,如果 `a` 为虚值则直接返回 `a`
+```js
+a || (b * c);  // 首先对 `a` 求值,如果 `a` 为真值则直接返回 `a`
+a && (b < c);  // 首先对 `a` 求值,如果 `a` 为虚值则直接返回 `a`
 a ?? (b || c); // 首先对 `a` 求值,如果 `a` 不是 `null` 或 `undefined` 则直接返回 `a`
 a?.b.c;        // 首先对 `a` 求值,如果 `a` 是 `null` 或 `undefined` 则直接返回 `undefined`
-
+``` -

示例

+## 示例 -
3 > 2 && 2 > 1
+```js
+3 > 2 && 2 > 1
 // 返回 true
 
-3 > 2 > 1
-// 返回 false,因为 3 > 2 是 true,然后 true 会在比较运算符中
-// 被隐式转换为 1,因此 true > 1 会变为 1 > 1,结果是 false
-// 加括号可以更清楚:(3 > 2) > 1
-
+3 > 2 > 1 +// 返回 false,因为 3 > 2 是 true,然后 true 会在比较运算符中 +// 被隐式转换为 1,因此 true > 1 会变为 1 > 1,结果是 false +// 加括号可以更清楚:(3 > 2) > 1 +``` -

汇总表

+## 汇总表 -

下面的表格将所有运算符按照优先级的不同从高(21)到低(1)排列。

+下面的表格将所有运算符按照优先级的不同从高(19)到低(1)排列。 -

请注意,下表中故意不包含展开语法(Spread syntax) —— 原因可以引用Stack Overflow 上的一个回答,“展开语法不是一个运算符,因此没有优先级。它是数组字面量和函数调用(和对象字面量)语法的一部分。”

+请注意,下表中故意不包含[展开语法(Spread syntax)](/zh-CN/docs/Web/JavaScript/Reference/Operators/Spread_syntax) —— 原因可以引用[Stack Overflow 上的一个回答](https://stackoverflow.com/a/48656377),“[展开语法不是一个运算符](https://stackoverflow.com/q/44934828/1048572),因此没有优先级。它是数组字面量和函数调用(和对象字面量)语法的一部分。” @@ -197,19 +197,19 @@ a?.b.c; // 首先对 `a` 求值,如果 `a` 是 `null` 或 `undefined` - + - - + + - + @@ -229,23 +229,23 @@ a?.b.c; // 首先对 `a` 求值,如果 `a` 是 `null` 或 `undefined` - + - - + + - + - + @@ -263,11 +263,11 @@ a?.b.c; // 首先对 `a` 求值,如果 `a` 是 `null` 或 `undefined` - + - + @@ -287,13 +287,13 @@ a?.b.c; // 首先对 `a` 求值,如果 `a` 是 `null` 或 `undefined` - + - + @@ -307,7 +307,7 @@ a?.b.c; // 首先对 `a` 求值,如果 `a` 是 `null` 或 `undefined` - + @@ -317,7 +317,7 @@ a?.b.c; // 首先对 `a` 求值,如果 `a` 是 `null` 或 `undefined` - + @@ -331,7 +331,7 @@ a?.b.c; // 首先对 `a` 求值,如果 `a` 是 `null` 或 `undefined` - + @@ -357,7 +357,7 @@ a?.b.c; // 首先对 `a` 求值,如果 `a` 是 `null` 或 `undefined` - + @@ -375,50 +375,49 @@ a?.b.c; // 首先对 `a` 求值,如果 `a` 是 `null` 或 `undefined` - + - + - + - + - + - - + - - + + @@ -467,16 +466,6 @@ a?.b.c; // 首先对 `a` 求值,如果 `a` 是 `null` 或 `undefined` - - - - - - - - - - @@ -484,4 +473,4 @@ a?.b.c; // 首先对 `a` 求值,如果 `a` 是 `null` 或 `undefined` -
运算符
2119 {{jsxref("Operators/Grouping", "分组", "", 1)}} n/a(不相关) ( … )
20{{jsxref("Operators/Property_Accessors", "成员访问", "#Dot_notation", 1)}}18{{jsxref("Operators/Property_Accessors", "成员访问", "#点号表示法", 1)}} 从左到右 … . …
{{jsxref("Operators/Property_Accessors", "需计算的成员访问", "#Bracket_notation", 1)}}{{jsxref("Operators/Property_Accessors", "需计算的成员访问", "#方括号表示法", 1)}} 从左到右 … [ … ]
?.
1917 {{jsxref("Operators/new","new")}}(无参数列表) 从右到左 new …
18{{jsxref("Operators","后置递增","#Increment", 1)}}16{{jsxref("Operators","后置递增","#自增和自减", 1)}} n/a … ++
{{jsxref("Operators","后置递减","#Decrement", 1)}}{{jsxref("Operators","后置递减","#自增和自减", 1)}} … --
1715 逻辑非 (!) 从右到左 ! …- …
前置递增{{jsxref("Operators","前置递增","#自增和自减", 1)}} ++ …
前置递减{{jsxref("Operators","前置递减","#自增和自减", 1)}} -- …
await …
1614 幂 (**) 从右到左 … ** …
1513 乘法 (*) 从左到右 … * …… % …
1412 加法 (+) 从左到右 … + …… - …
1311 按位左移 (<<) 从左到右 … << …… >>> …
1210 小于 (<) 从左到右 … < …… instanceof …
119 相等 (==) 从左到右 … == …… !== …
108 按位与 (&) 从左到右 … & …
97 按位异或 (^) 从左到右 … ^ …
86 按位或 (|) 从左到右 … | …
75 逻辑与 (&&) 从左到右 … && …
64 逻辑或 (||) 从左到右 … || …
5 空值合并 (??) 从左到右 … ?? …
43 条件(三元)运算符 从右到左 … ? … : …
3赋值2赋值 从右到左 … = …
… ??= …
2{{jsxref("Operators/yield", "yield")}}从右到左yield …
{{jsxref("Operators/yield*", "yield*")}}yield* …
1 逗号 / 序列… , …
\ No newline at end of file + -- cgit v1.2.3-54-g00ecf