JS 运算符技巧

【本文会持续更新!】

1. 转换成数字

  • + 运算符

使用 +运算符可以把其他类型转换成数字类型,但在使用时要注意表达式的结构,避免被解析成字符串连接符

+'123';             // 123

+'123'+'456';       // "123456"
+'123'+(+'456');    // 579
  • ~~ 位运算符

按位非(~)实质上是 对数字求负,然后减1。详细的处理过程请看 w3school: 位运算符

~1;    // -2
~-1;   // 0

那么再次执行按位非即可取回原值:-(-n-1)-1=n

~~1;      // 1
~~'1';    // 1

需要注意的是,~~ 的方式只适合处理 32 位以下整数,若是浮点数会被取整

~~1.2;     // 1
~~-1.2;    // -1

+ 运算符相比,~~ 运算符会把 undefined 或者不能转换成数字的值处理成 0,在某些业务计算场景下这样的处理是有方便之处的。

+undefined;     // NaN
+'abc';         // NaN
~~undefined;    // 0
~~'abc';        // 0

实际上, + 运算符和 ~~ 运算符都相当于使用 Number() 函数来处理,所以我们仍然需要关注一些特殊值的转换:

+'123', ~~'123';            // 123
+undefined, ~~undefined;    // NaN
+null, ~~null;              // 0
+true, ~~true;              // 1
+false, ~~false;            // 0
+[1], ~~[1];                // 1

当被转换值本身就是数字类型时,我们需要 小心被当成八进制数处理

+010;      // 8
+'010';    // '10'

2. 转换成字符串

+ 运算符紧跟一个空字符串,就可以把其他类型转换成字符串类型。

1+'';            // "1"
undefined+'';    // "undefined"
null+'';         // "null"
true+'';         // "true"
[1,2]+'';        // "1,2"

当被转换对象存在 toString() 的原型方法时,这种转换方式相当于调用了 toString() 函数:

new Date()+'';            // "Thu May 16 2019 20:42:43 GMT+0800 (China Standard Time)"
new Date().toString();    // "Thu May 16 2019 20:42:43 GMT+0800 (China Standard Time)"

var o = {a: 1};
o+'';            // "[object Object]"
o.toString();    // "[object Object]"

3. 转换成布尔值

由于逻辑非(!)返回的一定是布尔值,所以通过双取反即可转换成布尔值。

!!0;            // false
!!undefined;    // false
!!'abc';        // true
!![]            // true
!!{}            // true

4. 短路求值

逻辑与(&&)和逻辑或(||)运算都是简便运算,即如果第一个运算数决定了结果,就不再计算第二个运算数,这就是短路求值。

利用短路求值可以大幅减少逻辑判断的代码量,但同时也会降低代码可读性。

var condition = true;

if (condition) {
  console.log('It is true');
}
condition && console.log('It is true');

if (!condition) {
  console.log('It is false');
}
condition || console.log('It is false');

逻辑与(&&)表达式会返回第一个与 false 相等的值,而逻辑非(||)表达式会返回第一个与 true 相等的值,都没有则返回最后一个运算数的值。

null && false;    // null
1 && 2 && 3;      // 3
1 || true;        // 1
0 || false;       // false

5. 浮点数取整

由于浮点数是不支持位运算的,所以在运算之前会把浮点数的小数部分去掉,也就相当于对浮点数进行了取整。

只需要满足位运算后不改变值的表达式,都可以视作快速取整的一种方式。

  • 按位或 |

整数与 0 进行位或运算时,整数值不变,可用于浮点数取整。取整行为取决于浮点数是正数还是负数,正数时作向下取整,负数时作向上取整

// 向下取整
Math.floor(12.3);    // 12
12.3|0;              // 12

// 向上取整
Math.ceil(-12.3);    // -12
-12.3|0;             // -12

需要注意位或运算取整和 Math.floor() 等取整函数在特殊值处理上的差异:

Math.floor(NaN);         // NaN
NaN|0;                   // 0

Math.floor(Infinity);    // Infinity
Infinity|0;              // 0
  • ~~ 运算符

~~ 运算符利用的就是两次位非运算(~)后取回原值的特性,取整行为与位或运算一致。

~~12.3;     // 12
~~-12.3;    // -12

6. 奇偶判断

我们通常用取模运算符(%)来判断奇偶性:

n % 2 === 1 ? 'n是奇数' : 'n是偶数';

当数字转成二进制表达时,判断奇偶性只需要看最后一位是 1(奇数)还是 0(偶数),所以我们可以通过与 1 进行按位与运算来判断奇偶性。

1 & 1;    // 1
2 & 1;    // 0

n & 1 ? 'n是奇数' : 'n是偶数';

7. 幂运算(ES7)

** 是 ES7 新增的幂运算符,详见 tc39提案

Math.pow(2, 3);    // 8
2**3;              // 8

8. 整数交换

异或(^)运算具有这样的性质:

  • 满足交换律 a \oplus b \oplus c = a \oplus c \oplus b
  • 满足结合律 (a \oplus b) \oplus c = a \oplus (b \oplus c)
  • 自反性 a \oplus a = 0, a \oplus 0 = a

所以通过三次异或运算可以完成两个整数值(AB)的交换:

A = A \oplus B
B = A \oplus B = (A \oplus B) \oplus B = A
A = A \oplus B = (A \oplus B) \oplus A = B

var a = 1;
var b = 2;

a = a ^ b;    // 3
b = a ^ b;    // 1
a = a ^ b;    // 2

注意,异或运算不适合于浮点数及其他基本类型的变量交换。

除此之外,利用自反性异或运算还可以用于整数值的比较:运算结果为0则等值,非0则不等值。

1^1 = 0;
1^2 = 3;

9. void 0

很多 JS 框架、类库都会出现 void 0 这样的写法。void 运算符会对给定的表达式求值并返回 undefined,所以

void 0 === undefined;    // true

void 0 来替代 undefined 主要出于两点考虑:

  • 节省代码的字节数;
  • undefined 可以被重写,而 void作为 JS 的关键字不能被重写;

ps: undefined 在 ES5 中已经是全局对象的一个只读属性,但在局部作用域下依然能被局部变量覆盖。

undefined = 1;
console.log(undefined);      // undefined

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