在ECMAScript中,有非常丰富的运算符和操作符,在这篇文章中将按通常的分类来稍微整理一下,不过在整理之前,先说明一下:
1、虽然标题是运算符和操作符,然而在我看来并没有多少严格区分的必要,在英文中,貌似也是用一个Operator来表示,所以在下文中我可能会混用。甚至,一些不属于运算符和操作符范畴的,我也整理在这里,只要我觉得必要。
2、对于运算符的优先级,你无需一一牢记——我相信你知道最简单的”先乘除,后加减”,至于其它的,如果你不确定,加上括号好了。在ECMAScript中,优先级相同的从左向右运算。
3、对于一些编程语言通用的运算符,比如常用算术运算符(+-*/),我只会简单的列举一下,不会展开,但是请注意,并不是说这些不重要,相反,这些通用运算符甚至处于一个非常基础的地位,只是我觉得你应该早已经熟悉,没必要在这里花时间强调。
4、那么,这里重点关注什么呢?就是一些在ECMAScript中比较特殊的操作符,或者我认为值得花时间强调的一些地方。
运算符与操作符
类别 | 操作符 | 描述 | 说明 |
一元操作符 | ++ | 自增1 |
1、自增(减)有前置和后置两种类型,前置先自增(减)再参与其它运算,后置先参与其它运算再自增(减)。 2、在ES中,自增(减)不仅适用于整数,它们可以作用于任意值,对于不是Number类型的值,会先按前一篇文章中的规则隐式转换为Number,然后再自增(减),此时变量类型也会变成Number类型。 |
-- | 自减1 | ||
+ | 一元加 | 一元加最主要的应用就是将操作数转变为Number类型,相当于调用Number()转换。 | |
- | 一元减 | 一元减则是在一元加的基础之上再取其相反数。 | |
算术操作符 | + | 加 |
1、除了加(+)之外,如果操作数不是Number类型,会自动调用Number()转换为Number类型再进行计算。 2、对于加减(+-),除了作为算术运算符。还可以作为一元操作符(见上)。当然,由于字符串操作中对加号(+)的重载,还可以用于将任意数值(的字符串)相连,这也是第1点中为什么要除了加(+),它在含有非Number类型值时,会将所有操作数转换为字符串相连接。 3、与一般类C语言不同,在ES中,除(/)和取模(%)并不会区分整数和浮点数,比如 5 / 2 = 2.5 而不是2,5.3 % 3 = 2.3 而不是2。 4、任意运算,只要操作数含NaN,结果就是NaN。但并不是结果为NaN就一定有一个操作数为NaN,比如0/0也返回NaN。 5、对于含无穷Infinity的运算,规定比较多,这里就不列举了,可以参考原书,或者自行测试。 |
- | 减 | ||
* | 乘 | ||
/ | 除 | ||
% | 取模 | ||
逻辑操作符
(布尔操作符) |
! | 逻辑非 |
首先将操作数转换为Boolean类型值,然后再取反。可以使用双重非!!将一个数值转换为相应的Boolean值。 |
&& | 逻辑与 |
1、当两个操作数相应的Boolean值均为true时,返回true 2、短路:当第一个操作数相应的Boolean值为false时,会直接返回false,不会再计算第二个操作数。这常常被应用在判断一个变量(属性)是否有定义,如: |
|
|| | 逻辑或 |
1、当两个操作数相应的Boolean值至少有一个为true时,返回true 2、短路:当第一个操作数相应的Boolean值为true时,会直接返回true,不会再计算第二个操作数。 3、逻辑或,除了用于一般的判断之外,还常常被应用在提供默认值的情况,如: function Fn(obj){obj = obj || {}; } 这里如果调用Fn未传入obj,则会自动给obj赋值为undefined,然后因为undefined的相应Boolean值为false,所以会将一个空对象{}赋值给obj,如果调用传入了obj,则因为任意对象的Boolean值为true,所以就不会取后面的{},从而达到给obj一个默认值{}的效果。 这种方式还被应用在大型JS库的多个相对独立的文件中: //file2
|
|
关系操作符 (比较操作符) |
< | 小于 |
1、只要有一个操作数是Number类型或Boolean类型值,就将两个操作数转换成Number类型值(如果需要转换)执行数值比较。 2、字符串比较,会逐个比较字符编码值。 3、操作符是对象时,调用valueOf()(如果没有,就调用toString()),再将结果按上面规则比较。 4、任意数和NaN比较返回false。 |
<= | 小于或等于 | ||
> | 大于 | ||
>= | 大于或等于 | ||
== | 相等 |
1、相等和不等(==、!=)在比较时,只要有必要,就会隐式类型转换。 2、全等和不全等(===、!==)在比较时,不会转换类型,如果类型不一致,直接为!==。 3、结合1、2,可以知道,a===b则一定有a==b,而a!=b则一定有a!==b。 |
|
!= | 不等 | ||
=== | 全等 | ||
!== | 不全等 | ||
賦值運算子 | = | 賦值 | |
複合算術賦值運算子 | 算術運算子加上= | 對應算術運算符,有 =、-=、*=、/=、%= | |
複合位元賦值運算子 | 位元運算子加= | 對應位元運算符,有~=、&=、|=、^=、>=、>>>= | |
位元運算子 | ~ | 按位非 | 按位取反,也即回傳反碼 |
& | 按位與 | 按位對齊,逐位操作,只有兩個操作位均為1才回傳1,否則該位回傳0,最後將所有位操作結果組合回傳 | |
| | 按位或 | 按位對齊,逐位操作,只有兩個操作位均為0才返回0,否則該位返回1,最後將所有位操作結果組合回 | |
^ | 按位異或 | 按位對齊,逐位操作,兩個操作位不相同時返回1,否則該位返回0,最後將所有位操作結果組合返回 | |
左移 | 二進位數向左移位,左移不會改變符號位元 | ||
>> | 有符號右移 | 二進位數向右移位,高位以符合位元填入 | |
>>> | 無符號右移 | 二進制數向右移位,直接右移,對於正數,結果和>>相同,對於負數,會把負數的二進制補碼當成正數的二進制碼處理 | |
字串運算子 | 字串連接 | 相當於concat()函數,會先將所有運算元轉換為字串,然後再連接。注意,字串一旦創建就不會變更,執行字串連接時,在後台會有一個中間的連接和銷毀過程,這也是老舊瀏覽器在大量字串連接操作時運行緩慢的原因。 | |
= | 字串連接複合 | a =b,相當於a=a b。 | |
物件運算子 | . | 屬性存取符 | 簡單的物件屬性存取符。 |
[] | 屬性或(類別)陣列存取 | 透過[],可以存取名稱是一個變數或含有特殊字元的屬性。 | |
new | 呼叫建構子建立物件 | 傳回一個新建立的對象,在建構函式內部的this被指向這個新建立的物件。 | |
delete | 变量、属性删除 | 删除属性(变量可以看成是全局对象或执行环境的一个属性)。 | |
void | 返回undefined。 | ||
in | 判断属性 | 对象属性或原型链上的属性。 | |
instanceof | 原型判断 | 比较同一个上下文中的对象是否为另一个对象的原型。 | |
其它操作符 | ?: | 条件操作符 | 语法;var value = exp ? trueExp : falseExp。 相当于var value; if(exp){ value = trueExp;}else{value = falseExp;} |
, | 逗号操作符 | 主要用于声明多个变量,这也是很多JS库的流行做法。例如:var num1=1,num2=2,num3=3; | |
() | 分组操作符 |
主要用途: 1、结合逗号操作符用于赋值。例如:var num = (5,1,4,8,0);这里num最后的值为0。 2、转换为表达式。比如eval('('+jsStr+')');又比如: |
|
typeof | 类型操作符 |
返回一个字符串值:Undefined类型—>'undefined'、Null类型—>'object'、Boolean类型—>'boolean'、Number类型—>‘number'、String—>'string'、内置Function对象的实例—>'function'、其它Object类型—>'object'。(有些浏览器实现略有不同) |
說明幾點:
1、這裡的分類並不十分嚴格,例如按位非(~)、邏輯非(!)、delete、void、typeof,都可以算是一元操作符,而自增( )在很多資料中也被歸類為算術操作符之中。我在整理時則主要參考原書分類,也兼顧自然性。
2、加號( )的用法比較靈活,需注意,特別是用於計算時,忽略了其中的字串,會很容易犯錯。
3、typeof一般用來判斷簡單資料類型,如果是物件類型,因為大部分回傳的都是object,沒有多大實際用處,而instanceof的判斷也需要滿足同一個上下文的條件,否則也會出錯,對於物件類別的判斷會在後面講述物件時再詳細說明另一種更為穩健的方法。
4、先看下面的代碼: