前言

使用 webpack 工具打包 React 项目编译后的代码,经常会出现以下片段:

v=rt((0,r.useState)(void 0),2),  

为什么 r.useState 方法,要用 (0,r.useState)(void 0) 的方式去执行呢?
r.useState(void 0) 的调用方式也是可运行的,为什么不这么写?

带着这个问题,下文将重点探讨 (0, function)(param) 的应用。

逗号操作符

对它的每个操数求值(从左到右),并返回最后一个操作数的值。

let x = 1;  
  
x = (x++, x);  
  
console.log(x);  
// expected output: 2  
  
x = (2, 3);  
  
console.log(x);  
// expected output: 3  

熟悉 逗号操作符 这个知识点后,对 (0,r.useState)(void 0) 的前半部分,也就能认识了。
(0,r.useState) 也即 r.userState 对应的方法实例。

举个例子:

r.userState = function(){   
    console.log('leon');  
};  

(0,r.useState) 对应的方法实例为:

function(){   
    console.log('leon');  
}  

this 的指向

在浏览器控制台运行以下代码:

(function() {  
  (0,eval)("var leon = 123"); // 通过 eval 创建全局变量 leon  
})();  
console.log(leon);   // 输出 123  

eval 执行的代码环境上下文,通常是局部上下文。
但是,我们发现 eval 执行的代码 var leon = 123 在全局上下文中起效。

以下为 MDN 文档中,对 eval 作用域的说明:

function test() {  
  var x = 2, y = 4;  
  console.log(eval('x + y'));  // 直接调用,使用本地作用域,结果是 6  
    
  var geval = eval; // 等价于在全局作用域调用  
  console.log(geval('x + y')); // 间接调用,使用全局作用域,throws ReferenceError 因为`x`未定义  
}  

可以得出:(0,eval) 属于间接调用,使用的是 全局作用域this 指向的是全局上下文

使用以下代码验证下:

(function() {  
  (0,eval)("console.log(this)");  
})();  

发现输出 this 指向的是 Window

小结

通过 (0, function)(param) 方式调用 functionfunctionthis 指向的是 Window (全局上下文)。

为什么不用 call / apply 指定全局上下文 window ?

比如:

function sayName(){  
    console.log(this.name);  
}  
  
sayName.apply({  
 name: "leon"  
}, [])  

apply 是可以指定 this 的。

但是,原型链上的 apply 方法,是可以修改的。
比如:

Function.prototype.apply = function(){  
    console.log('new apply');  
};  
  
function sayName(){  
    console.log(this.name);  
}  
  
sayName.apply({  
 name: "leon"  
}, [])  

apply 方法,被重写后,失去了指定 this 的能力。

所以,为什么不用 call / apply 指定全局上下文 window ?
就是为预防 call / apply 被篡改后,导致程序运行异常。

为什么逗号操作符用 0 ?

其实,用其他数字或者字符串也是没问题的。
至于为什么用 (0, function) ? 可以说是业界的默认规则。
如果硬要说个为什么,可能是 0 在二进制的物理存储方式上,占用的空间较小。

总结

相信读者阅读到这里,对 (0,r.useState)(void 0) 应该不再感到陌生。

(0, function)(param) 的使用提炼如下:

  • 运用逗号操作符,获取到 function 实例的地址
  • 改变 this 指向全局上下文

参考