极客时间已完结课程限时免费阅读

JavaScript语法(四):新加入的**运算符,哪里有些不一样呢?

JavaScript语法(四):新加入的**运算符,哪里有些不一样呢?-极客时间

JavaScript语法(四):新加入的**运算符,哪里有些不一样呢?

讲述:winter

时长14:55大小13.62M

你好,我是 winter。
上一节课我们已经给你介绍了表达式的一些结构,其中关于赋值表达式,我们讲完了它的左边部分,而留下了它右边部分,那么,我们这节课一起来详细讲解。
在一些通用的计算机语言设计理论中,能够出现在赋值表达式右边的叫做:右值表达式(RightHandSideExpression),而在 JavaScript 标准中,规定了在等号右边表达式叫做条件表达式(ConditionalExpression),不过,在 JavaScript 标准中,从未出现过右值表达式字样。
JavaScript 标准也规定了左值表达式同时都是条件表达式(也就是右值表达式),此外,左值表达式也可以通过跟一定的运算符组合,逐级构成更复杂的结构,直到成为右值表达式。
关于这块的知识,我们有时会看到按照运算符来组织的讲解形式。
这样讲解形式是因为:对运算符来说的“优先级”,如果从我们语法的角度来看,那就是“表达式的结构”。讲“乘法运算的优先级高于加法”,从语法的角度看就是“乘法表达式和加号运算符构成加法表达式”。
对于右值表达式来说,我们可以理解为以左值表达式为最小单位开始构成的,接下来我们就来看看左值表达式是如何一步步构成更为复杂的语法结构。

更新表达式 UpdateExpression

左值表达式搭配 ++ -- 运算符,可以形成更新表达式。
-- a;
++ a;
a --
a ++
更新表达式会改变一个左值表达式的值。分为前后自增,前后自减一共四种。
我们要注意一下,这里在 ES2018 中,跟早期版本有所不同,前后自增自减运算被放到了同一优先级。

一元运算表达式 UnaryExpression

更新表达式搭配一元运算符,可以形成一元运算表达式,我们看下例子:
delete a.b;
void a;
typeof a;
- a;
~ a;
! a;
await a;
它的特点就是一个更新表达式搭配了一个一元运算符。

乘方表达式 ExponentiationExpression

乘方表达式也是由更新表达式构成的。它使用**号。
++i ** 30
2 ** 30 //正确
-2 ** 30 //报错
我们看一下例子,-2 这样的一元运算表达式,是不可以放入乘方表达式的,如果需要表达类似的逻辑,必须加括号。
这里我们需要注意一下结合性,** 运算是右结合的,这跟其它正常的运算符(也就是左结合运算符)都不一样。
我们来看一个例子。
4 ** 3 ** 2
事实上,它是这样被运算的:
4 ** (3 ** 2)
而不是这样被运算的:
(4 ** 3) ** 2
我们来实际在代码中执行一下试试。最终结果是 262144, 而不是 4096。

乘法表达式 MultiplicativeExpression

到这里,我们进入了比较熟悉的表达式类型,乘方表达式可以构成乘法表达式,用乘号或者除号、取余符号连接就可以了,我们看看例子:
x * 2;
乘法表达式有三种运算符:
*
/
%
它们分别表示乘、除和取余。它们的优先级是一样的,所以统一放在乘法运算表达式中。

加法表达式 AdditiveExpression

加法表达式是由乘法表达式用加号或者减号连接构成的。我们看下例子:
a + b * c
加法表达式有加号和减号两种运算符。
+
-
这就是我们小学学的加法和减法的意思了。不过要注意,加号还能表示字符串连接,这也比较符合一般的直觉。

移位表达式 ShiftExpression

移位表达式由加法表达式构成,移位是一种位运算,分成三种:
<< 向左移位
>> 向右移位
>>> 无符号向右移位
移位运算把操作数看做二进制表示的整数,然后移动特定位数。所以左移 n 位相当于乘以 2 的 n 次方,右移 n 位相当于除以 2 取整 n 次。
普通移位会保持正负数。无符号移位会把减号视为符号位 1,同时参与移位:
-1 >>> 1
这个会得到 2147483647,也就是 2 的 31 次方,跟负数的二进制表示法相关,这里就不详细讲解了。
在 JavaScript 中,二进制操作整数并不能提高性能,移位运算这里也仅仅作为一种数学运算存在,这些运算存在的意义也仅仅是照顾 C 系语言用户的习惯了。

关系表达式 RelationalExpression

移位表达式可以构成关系表达式,这里的关系表达式就是大于、小于、大于等于、小于等于等运算符号连接,统称为关系运算。
<=
>=
<
>
instanceof
in
需要注意,这里的 <= 和 >= 关系运算,完全是针对数字的,所以 <= 并不等价于 < 或 ==。例如:
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 类型之间时使用,比如:
document.getElementsByTagName('input')[0].value == 100
在这个例子中,等号左边必然是 string,右边的直接量必然是 number,这样使用 == 就没有问题了。

位运算表达式

位运算表达式含有三种:
按位与表达式 BitwiseANDExpression
按位异或表达式 BitwiseANDExpression
按位或表达式 BitwiseORExpression。
位运算表达式关系比较紧密,我们这里放到一起来讲。
按位与表达式由按位与运算符(&)连接按位异或表达式构成,按位与表达式把操作数视为二进制整数,然后把两个操作数按位做与运算。
按位异或表达式由按位异或运算符(^)连接按位与表达式构成,按位异或表达式把操作数视为二进制整数,然后把两个操作数按位做异或运算。异或两位相同时得 0,两位不同时得 1。
异或运算有个特征,那就是两次异或运算相当于取消。所以有一个异或运算的小技巧,就是用异或运算来交换两个整数的值。
let a = 102, b = 324;
a = a ^ b;
b = a ^ b;
a = a ^ b;
console.log(a, b);
按位或表达式由按位或运算符(|)连接相等表达式构成,按位或表达式把操作数视为二进制整数,然后把两个操作数按位做或运算。
按位或运算常常被用在一种叫做 Bitmask 的技术上。Bitmask 相当于使用一个整数来当做多个布尔型变量,现在已经不太提倡了。不过一些比较老的 API 还是会这样设计,比如我们在 DOM 课程中,提到过的 Iterator API,我们看下例子:
var iterator = document.createNodeIterator(document.body, NodeFilter.SHOW_TEXT | NodeFilter.SHOW_COMMENT, null, false);
var node;
while(node = iterator.nextNode())
{
console.log(node);
}
这里的第二个参数就是使用了 Bitmask 技术,所以必须配合位运算表达式才能方便地传参。

逻辑与表达式和逻辑或表达式

逻辑与表达式由按位或表达式经过逻辑与运算符连接构成,逻辑或表达式则由逻辑与表达式经逻辑或运算符连接构成。
这里需要注意的是,这两种表达式都不会做类型转换,所以尽管是逻辑运算,但是最终的结果可能是其它类型。
比如:
false || 1;
这句将会得到结果 1。
false && undefined;
这句将会得到 undefined。
另外还有一点,就是逻辑表达式具有短路的特性,例如:
true || foo();
这里的 foo 将不会被执行,这种中断后面表达式执行的特性就叫做短路。

条件表达式 ConditionalExpression

条件表达式由逻辑或表达式和条件运算符构成,条件运算符又称三目运算符,它有三个部分,由两个运算符?:配合使用。
condition ? branch1 : branch2
这里需要注意,条件表达式也像逻辑表达式一样,可能忽略后面表达式的计算。这一点跟 C 语言的条件表达式是不一样的。
条件表达式实际上就是 JavaScript 中的右值表达式了 RightHandSideExpression,是可以放到赋值运算后面的表达式。

总结

今天我们讲解了表达式的右边部分,讲到了包括更新表达式、一元运算表达式、乘方表达式、乘法表达式、移位表达式等 14 种表达式。至此为止,我们已经讲全了表达式。你如果有不熟悉的地方,可以随时回头查阅。
留一个小任务,我们试着总结下 JavaScript 中所有的运算符优先级和结合性。例如:
分享给需要的人,Ta购买本课程,你将得18
生成海报并分享

赞 16

提建议

上一篇
JavaScript语法(三):什么是表达式语句?
下一篇
HTML语义:div和span不是够用了吗?
 写留言

精选留言(26)

  • AbyssKR
    2019-04-09
    “逻辑与表达式和逻辑或表达式”一节中,第二个例子 false && undefined; 的结果为 false
    共 6 条评论
    70
  • 阿成
    2019-04-08
    优先级由高到低: 1 () 2 ./[] 左 2 new MemberExpression Arguments 3 new MemberExpression 右 4 () 函数调用 左 5 ++/-- 6 +/-/~/!/delete/typeof/void/await 右 7 ** 右 8 *///% 左 9 +/- 左 10 << >> >>> 左 11 < > <= >= instance of in 左 12 == === != !== 左 13 & 左 14 ^ 左 15 | 左 16 && 左 17 || 左 18 ?: 右 19 = 右 20 , 左 不过MDN上的一份整理是这样的:https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Operators/Operator_Precedence 主要不同的地方集中在: 1. 函数调用与New Expression的优先级顺序(上一课文中有一句不是很理解:而 Call Expression 就失去了比 New Expression优先级高的特性,这是一个主要的区分) 2. ++/-- 前置和后置的优先级顺序(MDN上前置后置优先级是不同的) 3. 缺少了 yield/yield*/...这三个运算符
    展开
    共 1 条评论
    31
  • windliang
    2019-06-28
    -1 >> 1,-1 的补码是 1111 1111 ... 1111 1111 ,32 个 1,有符号右移,高位补 1,所以还是 1111 1111 ... 1111 1111 ,32 个 1,所以答案依旧是 -1。 -1 >>> 1,-1 的补码是 1111 1111 ... 1111 1111 ,32 个 1,无符号右移,高位补 0,所以是 0111 1111... 1111 1111 ,31 个 1,代表 2 ** 31 - 1= 2147483648 - 1 = 2147483647 。 「在 JavaScript 中,二进制操作整数并不能提高性能」我觉得原因就是 js 数字不管整数还是小数都用 IEEE 754 浮点数标准。所以没有所谓的补码之说,也只有在进行位操作的时候,js 进行转换而已。 之前总结了一篇补码的文章,分享一下. https://mp.weixin.qq.com/s/uvcQHJi6AXhPDJL-6JWUkw
    展开

    作者回复: 优秀。

    共 3 条评论
    20
  • Geeker
    2019-04-09
    标准里有些东西还是看不太懂,如果可以的话,希望老师在答疑的时候稍微讲解一下如何看懂标准
    7
  • 冬Don
    2019-11-14
    为什么 [undefined] == 0 是 true, [false] == 0是false呢

    作者回复: [undefined].toString(); [false].toString(); 这么一看就知道为什么了。 不过不同类型的双等号比较是JS里的糟粕,你学了也没什么用处。

    4
  • 依韵
    2019-07-19
    false && undefined 值为false
    共 2 条评论
    3
  • 白嗣
    2019-05-20
    false && undefined; // false
    3
  • Geeker
    2019-04-09
    评论不能发图片,我把不太明白的地方截图发到了语雀: https://cdn.nlark.com/yuque/0/2019/jpeg/119718/1554814331032-assets/web-upload/a51218b6-cd8a-4ac8-b71f-f8f6dbfeab36.jpeg 麻烦老师移步看一下,谢谢。
    3
  • Geeker
    2019-04-06
    老师是在带我们过 ECMAScript标准,讲解很详细,基本能读懂标准了🌝
    3
  • 梧桐
    2019-09-25
    “逻辑与表达式和逻辑或表达式”一节中,第二个例子 false && undefined; 的结果为 false
    2
  • K4SHIFZ
    2019-04-14
    老师,在ES5之前版本规范中,会提及JS的可执行代码分为全局、函数、Eval。但是在ES6之后版本规范中,再也不提及可执行代码的概念了,这是为什么呢?
    共 1 条评论
    2
  • 阿成
    2019-04-07
    文中提到: 按位或表达式由按位或运算符(|)连接相等表达式构成,按位或表达式把操作数视为二进制整数,然后把两个操作数按位做或运算。 这里的相等表达式应该是按位异或表达式吧? (抱歉小编,之前打错了...)
    展开
    2
  • 你好,阳光
    2021-03-30
    文稿中说一元更新表达式是一个更新表达式搭配了一个一元运算符。但是从给的例子如+a,-a这些看不出来,能否换一些更恰当的例子?
    1
  • 渴望做梦
    2019-07-31
    老师,这篇文章里面每个表达式介绍的时候都说是由上一个表达式构成的,对此我不是很理解,比如一元运算表达式是有更新表达式和一元运算符构成的,但是 typeof a 我怎么看也没看出来哪有更新表达式

    作者回复: a本身就是一个更新表达式

    共 2 条评论
    1
  • study
    2019-05-30
    ExponentiationExpression,这个应该是指数表达式,不是乘法表达式吧。
    共 1 条评论
    1
  • 温木
    2019-05-06
    左移 n 位相当于乘以 2 的 n 次方,右移 n 位相当于除以 2 取整 n 次。 异或运算有个特征,那就是两次异或运算相当于取消。所以有一个异或运算的小技巧,就是用异或运算来交换两个整数的值。
    2
  • 乃乎
    2019-04-29
    “字符串和 bool 都转为数字再比较” 这点很重要,也很对
    2
  • Geeker
    2019-04-09
    比如本章和前一章对应标准中的 “12 ECMAScript Language: Expression”这一章,但有些内容不是很好懂,可能和标准的行文结构和规范有关,希望老师有时间可以稍做调拨,谢谢!

    作者回复: ES2018写的比较类似伪代码了,我这边还是从前端的理解角度出发来讲的。 你具体哪里不懂,可以发出来看看。

    1
  • 周伟
    2022-02-26
    更新表达式的介绍部分有点迷,下面表达式中哪里有更新表达式;delete a.b;void a;typeof a;- a;~ a;! a;await a;
  • 胡永
    2021-05-12
    作业题目可以通过这个里面的看到答案,标准里通过语法规则表述的很明白:https://tc39.es/ecma262/#sec-update-expressions