你说的this是哪个this?

「这是我参与2022首次更文挑战的第9天,活动详情查看:2022首次更文挑战

前言

我们都知道js里的this指向一直都是个令人头大的问题。在不同的环境,甚至是不同的语义、作用域里面this的指向千差万别。

比如在es6之前,我们在函数里面获取到的this,是指向函数自身的,修改了外部作用域的this指向;es6之后,我们使用箭头函数,避免了函数改变this指向的问题。

常用来修改this指向的方法还有 bind、apply、call。三者第一个参数都是this, bind返回的函数不会立即执行,apply 第二个参数是数组。bind和call的参数都是散列传递的,即从第二个往后都是传递给函数的参数,逗号隔开。

globalThis 对象

JavaScript 语言存在一个顶层对象,它提供全局环境(即全局作用域),所有代码都是在这个环境中运行。但是,顶层对象在各种实现里面是不统一的。

  • 浏览器里面,顶层对象是window,但 Node 和 Web Worker 没有window
  • 浏览器和 Web Worker 里面,self也指向顶层对象,但是 Node 没有self
  • Node 里面,顶层对象是global,但其他环境都不支持。

同一段代码为了能够在各种环境,都能取到顶层对象,现在一般是使用this关键字,但是有局限性。

  • 全局环境中,this会返回顶层对象。但是,Node.js 模块中this返回的是当前模块,ES6 模块中this返回的是undefined
  • 函数里面的this,如果函数不是作为对象的方法运行,而是单纯作为函数运行,this会指向顶层对象。但是,严格模式下,这时this会返回undefined
  • 不管是严格模式,还是普通模式,new Function('return this')(),总是会返回全局对象。但是,如果浏览器用了 CSP(Content Security Policy,内容安全策略),那么evalnew Function这些方法都可能无法使用。

综上所述,很难找到一种方法,可以在所有情况下,都取到顶层对象。下面是两种勉强可以使用的方法。

// 方法一
(typeof window !== 'undefined'
   ? window
   : (typeof process === 'object' &&
      typeof require === 'function' &&
      typeof global === 'object')
     ? global
     : this);

// 方法二
var getGlobal = function () {
  if (typeof self !== 'undefined') { return self; }
  if (typeof window !== 'undefined') { return window; }
  if (typeof global !== 'undefined') { return global; }
  throw new Error('unable to locate global object');
};
复制代码

ES2020 在语言标准的层面,引入globalThis作为顶层对象。也就是说,任何环境下,globalThis都是存在的,都可以从它拿到顶层对象,指向全局环境下的this

image.png

垫片库global-this模拟了这个提案,可以在所有环境拿到globalThis

总结

js设计之初存在的问题,会逐渐在后面的es标准中逐渐完善和修复,我们在享受新语法带来的快感的同时,也应该尽其所能为这门语言的发展出谋划策、贡献力量。

猜你喜欢

转载自juejin.im/post/7063047546205110279