You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

320 lines
10 KiB
Markdown

This file contains ambiguous Unicode characters!

This file contains ambiguous Unicode characters that may be confused with others in your current locale. If your use case is intentional and legitimate, you can safely ignore this warning. Use the Escape button to highlight these characters.

# JavaScript语法新加入的**运算符,哪里有些不一样呢?
你好我是winter。
上一节课我们已经给你介绍了表达式的一些结构,其中关于赋值表达式,我们讲完了它的左边部分,而留下了它右边部分,那么,我们这节课一起来详细讲解。
在一些通用的计算机语言设计理论中能够出现在赋值表达式右边的叫做右值表达式RightHandSideExpression而在JavaScript标准中规定了在等号右边表达式叫做条件表达式ConditionalExpression不过在JavaScript标准中从未出现过右值表达式字样。
JavaScript标准也规定了左值表达式同时都是条件表达式也就是右值表达式此外左值表达式也可以通过跟一定的运算符组合逐级构成更复杂的结构直到成为右值表达式。
关于这块的知识,我们有时会看到按照运算符来组织的讲解形式。
这样讲解形式是因为:对运算符来说的“优先级”,如果从我们语法的角度来看,那就是“表达式的结构”。讲“乘法运算的优先级高于加法”,从语法的角度看就是“乘法表达式和加号运算符构成加法表达式”。
对于右值表达式来说,我们可以理解为以左值表达式为最小单位开始构成的,接下来我们就来看看左值表达式是如何一步步构成更为复杂的语法结构。
## 更新表达式 UpdateExpression
左值表达式搭配 `++` `--` 运算符,可以形成更新表达式。
```JavaScript
-- a;
++ a;
a --
a ++
```
更新表达式会改变一个左值表达式的值。分为前后自增,前后自减一共四种。
我们要注意一下这里在ES2018中跟早期版本有所不同前后自增自减运算被放到了同一优先级。
## 一元运算表达式 UnaryExpression
更新表达式搭配一元运算符,可以形成一元运算表达式,我们看下例子:
```JavaScript
delete a.b;
void a;
typeof a;
- a;
~ a;
! a;
await a;
```
它的特点就是一个更新表达式搭配了一个一元运算符。
## 乘方表达式 ExponentiationExpression
乘方表达式也是由更新表达式构成的。它使用`**`号。
```JavaScript
++i ** 30
2 ** 30 //正确
-2 ** 30 //报错
```
我们看一下例子,-2这样的一元运算表达式是不可以放入乘方表达式的如果需要表达类似的逻辑必须加括号。
这里我们需要注意一下结合性,\*\*运算是右结合的,这跟其它正常的运算符(也就是左结合运算符)都不一样。
我们来看一个例子。
```JavaScript
4 ** 3 ** 2
```
事实上,它是这样被运算的:
```JavaScript
4 ** (3 ** 2)
```
而不是这样被运算的:
```JavaScript
(4 ** 3) ** 2
```
我们来实际在代码中执行一下试试。最终结果是262144 而不是4096。
## 乘法表达式 MultiplicativeExpression
到这里,我们进入了比较熟悉的表达式类型,乘方表达式可以构成乘法表达式,用乘号或者除号、取余符号连接就可以了,我们看看例子:
```JavaScript
x * 2;
```
乘法表达式有三种运算符:
```
*
/
%
```
它们分别表示乘、除和取余。它们的优先级是一样的,所以统一放在乘法运算表达式中。
## 加法表达式 AdditiveExpression
加法表达式是由乘法表达式用加号或者减号连接构成的。我们看下例子:
```JavaScript
a + b * c
```
加法表达式有加号和减号两种运算符。
```
+
-
```
这就是我们小学学的加法和减法的意思了。不过要注意,加号还能表示字符串连接,这也比较符合一般的直觉。
## 移位表达式 ShiftExpression
移位表达式由加法表达式构成,移位是一种位运算,分成三种:
```
<< 向左移位
>> 向右移位
>>> 无符号向右移位
```
移位运算把操作数看做二进制表示的整数然后移动特定位数。所以左移n位相当于乘以2的n次方右移n位相当于除以2取整n次。
普通移位会保持正负数。无符号移位会把减号视为符号位1同时参与移位
```
-1 >>> 1
```
这个会得到2147483647也就是2的31次方跟负数的二进制表示法相关这里就不详细讲解了。
在JavaScript中二进制操作整数并不能提高性能移位运算这里也仅仅作为一种数学运算存在这些运算存在的意义也仅仅是照顾C系语言用户的习惯了。
## 关系表达式 RelationalExpression
移位表达式可以构成关系表达式,这里的关系表达式就是大于、小于、大于等于、小于等于等运算符号连接,统称为关系运算。
```
<=
>=
<
>
instanceof
in
```
需要注意,这里的<= 和 >= 关系运算,完全是针对数字的,所以 <= 并不等价于 < 或 ==。例如:
```JavaScript
null <= undefined
//false
null == undefined
//true
```
请你务必不要用数学上的定义去理解这些运算符。
## 相等表达式 EqualityExpression
在语法上,相等表达式是由关系表达式用相等比较运算符(如 `==`)连接构成的。所以我们可以像下面这段代码一样使用,而不需要加括号。
```
a instanceof "object" == true
```
相等表达式由四种运算符和关系表达式构成,我们来看一下运算符:
* `==`
* `!=`
* `===`
* `!==`
相等表达式又包含一个JavaScript中著名的设计失误那就是 `==` 的行为。
一些编程规范甚至要求完全避免使用 `==` 运算,我觉得这样规定是比较合理的,但是这里我还是尽量解释一下 `==` 的行为。
虽然标准中写的`==`十分复杂,但是归根结底,类型不同的变量比较时`==`运算只有三条规则:
* undefined与null相等
* 字符串和bool都转为数字再比较
* 对象转换成primitive类型再比较。
这样我们就可以理解一些不太符合直觉的例子了,比如:
* `false == '0'` true
* `true == 'true'` false
* `[] == 0` true
* `[] == false` true
* `new Boolean('false') == false` false
这里不太符合直觉的有两点:
* 一个是即使字符串与boolean比较也都要转换成数字
* 另一个是对象如果转换成了primitive类型跟等号另一边类型恰好相同则不需要转换成数字。
此外,`==` 的行为也经常跟if的行为转换为boolean混淆。总之我建议仅在确认 `==` 发生在Number和String类型之间时使用比如
```JavaScript
document.getElementsByTagName('input')[0].value == 100
```
在这个例子中等号左边必然是string右边的直接量必然是number这样使用 == 就没有问题了。
## 位运算表达式
位运算表达式含有三种:
* 按位与表达式 BitwiseANDExpression
* 按位异或表达式 BitwiseANDExpression
* 按位或表达式 BitwiseORExpression。
位运算表达式关系比较紧密,我们这里放到一起来讲。
按位与表达式由按位与运算符(`&`)连接按位异或表达式构成,按位与表达式把操作数视为二进制整数,然后把两个操作数按位做与运算。
按位异或表达式由按位异或运算符(`^`连接按位与表达式构成按位异或表达式把操作数视为二进制整数然后把两个操作数按位做异或运算。异或两位相同时得0两位不同时得1。
异或运算有个特征,那就是两次异或运算相当于取消。所以有一个异或运算的小技巧,就是用异或运算来交换两个整数的值。
```JavaScript
let a = 102, b = 324;
a = a ^ b;
b = a ^ b;
a = a ^ b;
console.log(a, b);
```
按位或表达式由按位或运算符(`|`)连接相等表达式构成,按位或表达式把操作数视为二进制整数,然后把两个操作数按位做或运算。
按位或运算常常被用在一种叫做Bitmask的技术上。Bitmask相当于使用一个整数来当做多个布尔型变量现在已经不太提倡了。不过一些比较老的API还是会这样设计比如我们在DOM课程中提到过的Iterator API我们看下例子
```JavaScript
var iterator = document.createNodeIterator(document.body, NodeFilter.SHOW_TEXT | NodeFilter.SHOW_COMMENT, null, false);
var node;
while(node = iterator.nextNode())
{
console.log(node);
}
```
这里的第二个参数就是使用了Bitmask技术所以必须配合位运算表达式才能方便地传参。
## 逻辑与表达式和逻辑或表达式
逻辑与表达式由按位或表达式经过逻辑与运算符连接构成,逻辑或表达式则由逻辑与表达式经逻辑或运算符连接构成。
这里需要注意的是,这两种表达式都不会做类型转换,所以尽管是逻辑运算,但是最终的结果可能是其它类型。
比如:
```JavaScript
false || 1;
```
这句将会得到结果 1。
```JavaScript
false && undefined;
```
这句将会得到undefined。
另外还有一点,就是逻辑表达式具有短路的特性,例如:
```JavaScript
true || foo();
```
这里的foo将不会被执行这种中断后面表达式执行的特性就叫做短路。
## 条件表达式 ConditionalExpression
条件表达式由逻辑或表达式和条件运算符构成,条件运算符又称三目运算符,它有三个部分,由两个运算符`?`和`:`配合使用。
```JavaScript
condition ? branch1 : branch2
```
这里需要注意条件表达式也像逻辑表达式一样可能忽略后面表达式的计算。这一点跟C语言的条件表达式是不一样的。
条件表达式实际上就是JavaScript中的右值表达式了 RightHandSideExpression是可以放到赋值运算后面的表达式。
## 总结
今天我们讲解了表达式的右边部分讲到了包括更新表达式、一元运算表达式、乘方表达式、乘法表达式、移位表达式等14种表达式。至此为止我们已经讲全了表达式。你如果有不熟悉的地方可以随时回头查阅。
留一个小任务我们试着总结下JavaScript中所有的运算符优先级和结合性。例如
![](https://static001.geekbang.org/resource/image/4c/ca/4cb75eb863d5dffe7e9b6b0fb1161aca.jpg)