JavaScript浅析 -- 运算符和优先级

一、运算符

JavaScript中的运算符,主要用于连接简单表达式,组成一个复杂的表达式。在运算过程中,会根据需要将操作数转换成特定的类型进行运算,最后输出指定的类型。我个人习惯将他分为6大类进行记忆:算术运算符,比较运算符,布尔运算符,赋值运算符,位运算符,其他运算符。

(一)算术运算符(10个)
  • 加法运算符(Addition):x + y
  • 减法运算符(Subtraction): x - y
  • 乘法运算符(Multiplication): x * y
  • 除法运算符(Division):x / y
  • 余数运算符(Remainder):x % y
  • 自增运算符(Increment):++x 或者 x++
  • 自减运算符(Decrement):--x 或者 x--
  • 求负运算符(Negate):-x
  • 数值运算符(Convert to number): +x
  • 指数运算符 (Exponential operator) : x**y ,如2**4=16

说明:
(1) x++先参与运算然后x再做加一操作。如console.log(2 - a++)相当于console.log(2-a); a = a+1;++x先进行加一操作再参与运算。如console.log(2 - ++a)相当于a=a+1;console.log(2-a)
(2) 取余的正负号取决于第一位数,如-7%2为-1;7%-2为1,所以为了得到负数的正确余值需要用绝对函数,如判断num是否为偶数时Math.abs(num) % 2
(3) +x一般用来做转数的操作,如+new Date()、+[] // 0、+{} // NaN、+true // 1,其实相当于Number(x)只是结果为正。
(4) +号运算时才确定是相加还是连接,如果有一方是字符串则变成连接符。对于对象相加先调用valueOf()再调用toString(),但是valueOf()返回的一般是原对象,所以一般都会调用toString()。特例,Date对象反着来,先toString()再valueOf()。如'3' + 5 + 6 // 3563 + 5 + '6' // ‘86’

(二)比较运算符(8个)
  • == :相等
  • === :严格相等
  • != :不相等
  • !== :严格不相等
  • < :小于
  • <= :小于或等于
  • > :大于
  • >= :大于或等于

说明:
===是严格等于,要求类型和数值都相等(其中NaN===NaN为false);而==会进行隐式转换之后再比较值是否相等。具体的转换规则参考文章

(三)布尔运算符(4个)
  • ! :取反运算符
  • && :且运算符
  • || :或运算符
  • condition? true case : false case :三元条件运算符

说明:
(1) !a会对a先做Boolean()转换操作,然后对转换的布尔值进行取反,如!0为!Boolean(0)结果为true。
(2) 且和或的运算均为短路运算,即a && b两个为true才为true,a为false则不再管b;a || b一个为true则为true,a为true则不再管b。
(3) condition ? a : b相当于if(condition) { a } else { b }

(四)位运算符(7个)
  • 或运算(or):符号为|,表示两个二进制位中有一个为1,则结果为1,否则为0。
  • 与运算(and):符号为&,表示两个二进制位都为1,则结果为1,否则为0。
  • 否运算(not):符号为~,表示将一个二进制位变成相反值。
  • 异或运算(xor):符号为ˆ,表示两个二进制位中有且仅有一个为1时,结果为1,否则为0。
  • 左移运算(left shift):符号为<<
  • 右移运算(right shift):符号为>>
  • 带符号位的右移运算(zero filled right shift):符号为>>>
(五)赋值运算符(11个)
  • x += y // 等同于 x = x + y
  • x -= y // 等同于 x = x - y
  • x *= y // 等同于 x = x * y
  • x /= y // 等同于 x = x / y
  • x %= y // 等同于 x = x % y
  • x >>= y // 等同于 x = x >> y
  • x <<= y // 等同于 x = x << y
  • x >>>= y // 等同于 x = x >>> y
  • x &= y // 等同于 x = x & y
  • x |= y // 等同于 x = x | y
  • x ^= y // 等同于 x = x ^ y
(六)其他运算符
  • 小括号:有两种用法,如果把表达式放在圆括号之中,作用是求值;如果跟在函数的后面,作用是调用函数。
  • void:void运算符的作用是执行一个表达式,然后返回undefined。常见的a标签写上javascript:void(0)就是这个原理,会运行冒号后面的表达式但返回undefined就没执行。
  • 逗号运算符:逗号运算符用于对两个表达式求值,并返回后一个表达式的值。
  • typeof运算符:返回操作数的类型,其中typeof null结果'object',typeof Function结果是'function'。
  • instanceof运算符:a instanceof b判断a是否是b的实例,返回布尔值。

二、运算符的优先级

运算符的优先级顺序如下(上大于下,左大于右)
++、--(右结合大于左结合)、-(转负数)、+(转数)、~(位反)、!(取反)
delete、typeof、void
*、/、%、+、-
<=、>=、==、!=、===、!==
||、&&、?:、赋值、,

说明:
(1) 优先级指的是同时出现的时候先计算高优先级部分,而不是说先当做这个。例如转数的加号优先级高于加法的加号,1+'3‘++new Date()不是说把加号先当做转数符,如果是那么1+'3'结果是4但结果是'13‘,很显然是作为加号使用。而是说先计算+new Date()得到时间戳1531139346042,再执行1+'3‘+1531139346042
(2) ++、--之类的操作数必须是可以放在赋值左边的数而不能是常数,比如4++会报错。还有下面的例子也要注意:

var a=0, b=0;
a+++b; // 0
a; // 1
b; // 0

右结合的优先级高于左结合和加号,上面会先找有没有右结合,找到a++先运算,所以运算顺序是(a++)+b
(3) typeof的优先级比加减乘除还高,所以一般习惯把操作数加上括号,如

typeof 2*3; // NaN
typeof (2*3); // 'number'
typeof 2+3; // 'number3'

(4) 逻辑非优先级高于加减乘除,但逻辑或和逻辑与比加减乘除优先级低。!2*0相当于(!2)*0
(5) 逻辑或||、逻辑与&&的短路运算要注意,且返回的是其中的一个表达式,而不是布尔值。1 && "hi" || 0的结果是'hi'
(6) 赋值的优先级非常低。

a = b == c // 相当于a = (b==c)

下面这个例子,先自己猜一下结果,再看正确答案。arr.length会报错吗?

var arr, arr2;
arr2 = arr = [1, 2, 3];
arr[arr.length] = arr = 4;
console.log(arr);
console.log(arr2);

上面这个,由于赋值的优先级非常低最后才执行,arr[arr.length] = arr = 4其实是先算的arr[arr.length]是指向arr[3]的一个地址(和arr2[3]指向的是同一个),接着4赋值给arr然后arr的值4赋值给这块地址指向的内存(也就是arr2[3]变成了4)。所以结果是arr为4,arr2为[1, 2, 3, 4]。
再来看个例子也是类似。

var a = {n: 1};
var b = a;
a.x = a = {n:2};
console.log(a.x); // undefined
console.log(b.x); // {n: 2}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,723评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,003评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,512评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,825评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,874评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,841评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,812评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,582评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,033评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,309评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,450评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,158评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,789评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,409评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,609评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,440评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,357评论 2 352

推荐阅读更多精彩内容

  • 第2章 基本语法 2.1 概述 基本句法和变量 语句 JavaScript程序的执行单位为行(line),也就是一...
    悟名先生阅读 4,145评论 0 13
  • 运算符是处理数据的基本方法,用来从现有的值得到新的值。JavaScript 提供了多种运算符,本章逐一介绍这些运算...
    许先生__阅读 602评论 0 3
  • 运算符是处理数据的基本方法,用来从现有的值得到新的值。JavaScript 提供了多种运算符,本章逐一介绍这些运算...
    徵羽kid阅读 676评论 0 0
  • 这几天自己一直在和老板娘沟通品项事情,开始沟通的时候会有障碍,因为老板娘会回馈觉得不够系统化,她听的迷糊,你说的迷...
    zmy朱朱阅读 494评论 0 1
  • #清凉法语# 肚量多大事业就有多大 肚量多大,事业就有多大;要空出所有,才能建设一切。从管理上来说,我们的心能包容...
    xcy无名阅读 468评论 0 1