javaScript中的顶层对象小结

顶层对象,在浏览器中指的是window,在node环境中指的是global对象.

在ES5环境中,通过var声明的变量能够通过window对象获取到,

variable = '1';
console.log(window.variable);//1
window.variable = '2';
console.log(variable);//2

证明顶层对象的属性赋值与全局变量的赋值,是同一件事。这种特性是javaScript中广受吐槽的原因之一.这回带来以下几个问题:

1.如果变量因为未声明,那么只有在程序运行时才知道,无法在编译时就知道,因为全局变量可能是顶层对象的属性创造的,而属性的创造是动态的.

2.其次,程序员很容易不知不觉地就创建了全局变量(比如打字出错)

3.顶层对象的属性是到处可以读写的,这非常不利于模块化编程

4.window在浏览器环境中指的是窗口对象.有实体含义,顶层对象是一个有实体含义的对象,显然不合适的。

在ES6中为了解决顶层对象属性与全局变量混淆的情况,一方面规定,为了保持兼容性,var命令和function命令声明的全局变量,依旧是顶层对象的属性;另一方面规定,let命令、const命令、class命令声明的全局变量,不属于顶层对象的属性。也就是说,从 ES6 开始,全局变量将逐步与顶层对象的属性脱钩。

    var variable = '1';
    console.log(window.variable);//1
    let variable1 = '2';
    console.log(window.variable1);//undefined

各种环境中顶层对象的区分

1.浏览器里面,顶层对象是window,但 Node 和 Web Worker 没有window

2.浏览器和 Web Worker 里面,self也指向顶层对象,但是 Node 没有self

3.Node 里面,顶层对象是global,但其他环境都不支持。

还有一种方法是在不同的环境中,可以使用this来获取顶层对象,但是这种方法是有缺陷的

1.全局环境中,this会返回顶层对象。但是,Node 模块和 ES6 模块中,this返回的是当前模块。

2.函数里面的this,如果函数不是作为对象的方法运行,而是单纯作为函数运行,this会指向顶层对象。但是,严格模式下,这时this会返回undefined

3.不管是严格模式,还是普通模式,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');
    };

现在有一个提案,在语言标准的层面,引入global作为顶层对象。也就是说,在所有环境下,global都是存在的,都可以从它拿到顶层对象。垫片库system.global模拟了这个提案,可以在所有环境拿到global




参考:阮一峰ECMAScript 6

猜你喜欢

转载自blog.csdn.net/qq_36725757/article/details/80053769