JavaScript 尽量少用全局变量

尽量少用全局变量

JavaScript 使用函数管理作用域。变量在函数内声明,只在函数内有效,不能在外部使用。全局变量与之相反,在函数外部声明,在函数内无需声明即可简单地使用。

  • 每一个 JavaScript 环境都有全局对象,可以在函数外部使用 this 进行访问。创建的每一个全局变量都为全局对象所有。
var name = 'spirit';
console.log(this);      // Window 对象,并且该对象上多了一个 name 属性
console.log(this.name); // spirit
  • 在浏览器中,为了方便,使用 window 表示全局对象本身。
console.log(window);            // window 对象
console.log(this);              // window 对象
console.log(window === this);   // true

// 创建全局变量的方式:
// 方式一:在函数外声明变量
var global1 = "global1";
// 方式二:函数内不使用 var 关键字声明变量
function func() {
    global2 = "global2";
}
func();
console.log(window);             // window 对象,并且该对象上多了 global1、global2 两个属性
console.log(window.global1);     // global1
console.log(window['global2']);  // global2

全局变量的问题:

问题在于全局变量在整个 JavaScript 应用或 Web 页面内共享。它们生存于同一个全局命名空间内,总有可能发生命名冲突。譬如当一个应用程序中两个独立的部分定义了同名的全局变量,但却有不同目的时。

[图片上传失败...(image-373c91-1547520893760)]

  • 情境:某个第三方脚本定义了一个全局变量 result,然而开发人员不知道这个情况,在某个函数里定义了另一个全局变量,也叫做 result。这样造成的后果是后一个 result 变量覆盖前一个。如果开发人员定义的 result 在后面,那么第三方脚本可能会报错,甚至停止工作。
  • 结论:在同一个页面(命名空间内)尽可能少地使用全局变量
  • 根源:开发人员总是会在不知不觉中创建全局变量,其原因在于 JavaScript 的两个特性。
    • 直接使用变量,无需声明:在函数内部不使用 var 关键字声明的变量,默认为全局变量。更通俗地讲,任何变量,如果未经声明,就为全局变量。
    • 隐式全局变量:var 声明的链式赋值,例子如下:
function func() {
    var a = b = 0;
    /*
     * a 为局部变量,b 为全局变量
     * 产生的原因很简单:从右到左的操作符优先级。先执行 b = 0,而此刻 b 未声明,
     * 因此 b 为全局变量。整个过程等价于:
     * var a;
     * b = 0;
     * a = b;
     */
}
// 正确的做法:
function func() {
    var a, b;
    a = b = 0;
}
  • 方法
    1. 无论是全局变量还是局部变量,都用 var 声明变量。这么做的好处是,局部变量不会因为根源一成为全局变量。同时明确了全局变量与局部变量的界限,即通过变量声明的位置来判断变量的类型。
    2. 声明变量的同时赋初值,且尽量单独赋值。使用链式赋值,切记先对链式赋值的所有变量先进行声明。

变量释放时的副作用

隐含全局变量(不使用 var 关键字创建)与明确定义的全局变量有细微的不同。不同之处在于能否使用 delete 操作符撤销变量。

  • 使用 var 关键字声明的全局变量,不能删除。
  • 不使用 var 关键字声明的隐含全局变量,可以删除。

这表明隐含全局变量严格来讲不是真正的变量,而是全局对象的属性。属性可以通过 delete 操作符删除,而变量不行。

// 明确定义的全局变量
var global1 = "global1";
// 隐含全局变量
function func() {
    global2 = "global2";
}
func();
// 删除这两个全局变量
delete global1;
delete global2;

// 测试删除情况
console.log(typeof global1);    // string
console.log(typeof global2);    // undefined

访问全局对象

在浏览器下,可通过 window 属性在代码的任意位置访问到全局对象(除非做了特别处理而发生了意外,如声明了一个名为 window 的局部对象)。

  • 存在的问题:在其他环境下,全局对象的标识可能不叫 window,而是其他名字,甚至可能对于程序员是看不见的。那么在这种情况下,如何访问全局对象呢?

  • 解决的方法:从内嵌函数的作用域访问。按这种通常能获得全局对象,因为 this 在函数内部作为一个函数调用(不通过构造器 new 创建)时,往往指向该全局对象。

var global = (function() {
    return this;
}());
console.log(global);    // window 对象
  • 注意事项:在 ECMAScript 5 的严格模式下不能这么做。

单一 var 模式(Single var Pattern)

使用一个 var 关键字在函数顶部用逗号进行多个变量声明。

好处:

  • 提供一个单一的地址以查找到函数需要的所有局部变量。
  • 防止出现变量在定义(赋值)前就被使用的逻辑错误。
  • 帮助牢记要声明变量,以尽可能少地使用全局变量。
  • 更少的编码(无论是输入代码还是传输代码)。

示例:

function func() {
    var a = 1,
        b = 2,
        sum = a + b,
        myObject = {},
        i,
        j;
        
    // 函数体...
}

初始化变量

声明变量的同时初始化变量(为变量赋初值)。

好处:

  • 防止逻辑错误(所有未初始化且未声明的变量,其值都为 undefined)。
  • 提高代码可读性(当你在以后重新看到这段代码时,你可以根据变量的初始值知道使用这些变量的意图,比如该变量应该是对象还是数组?)。

提升:凌散变量的问题

提升:允许在函数的任意地方声明多个变量,无论在哪里声明,效果都等同于在函数顶部进行声明。

存在的问题:在变量声明前使用变量,会导致逻辑错误。因为变量提升,在同一个范围(同一函数)里声明的变量会默认提到函数顶部,因此无论在函数哪个位置使用该变量,该变量都可以视为已经声明。

[图片上传失败...(image-2a08f8-1547520893760)]

myName = "global";  // 全局变量
function func() {
    alert(myName);  // "undefined"
    var myName = "local";
    alert(myName);  // 局部变量
}
func();

// 前面代码片段运行结果等同于下述代码

myName = "global";      // 全局变量
function func() {
    var myName;         // 等同于 -> var myName = undefined
    alert(myName);      // undefined
    myName = "local";
    alert(myName);      // 局部变量
}

命名空间

简单的通过全局对象的单一属性表示的功能性分组。

  • 举例
    YUI 依照命名空间的思路来管理其代码,例如 Y.DOM 下的所有方法都是和 DOM 操作相关,Y.Event 下的所有方法都是和事件相关,以此类推。
  • 优点
    将功能按照命名空间进行分组,可以让你的单全局对象变得井然有序,同时可以让团队成员能够知晓新功能应该属于哪个部分,或者知道去哪里查找已有的功能。

创建命名空间的方法:

  • 对象字面量
    创建一个简单的对象字面量来打包所有的相关对象与函数,达到模拟命名空间的作用。
// 创建一个命名空间
var namespace = {};

// 在该命名空间上挂载一个名为 DOM 的对象
namespace.DOM = {};
// 在该命名空间上挂载一个名为 Event 的对象
namespace.Event = {};
  • 函数创建
    通过声明 function 实现,函数里设置初始变量,公共方法写入 prototype。

一.普通写法:

// 创建一个命名空间
var NameSpace = NameSpace || {};
// 在该命名空间上挂载一个名为 Hello 的构造函数
NameSpace.Hello = function() {
    this.name = 'world';
};
// 为 Hello 对象写入公共方法
NameSpace.Hello.prototype.sayHello = function(_name) {
    return 'Hello ' + (_name || this.name);
};
var hello = new NameSpace.Hello();

二.简洁写法:

var NameSpace = window.NameSpace || {};
NameSpace.Hello = new function() {
    var self = this,
        name = 'world';
    self.sayHello = function(_name) {
        return 'Hello' + (_name || name);
    };
};

不清楚这种方法为什么能够实现命名空间的作用,新建的 hello 变量仍然是个全局变量,如果在其他外部文件中也有 hello 变量,那么这不就造成命名冲突了?

  • 闭包和 Object 实现
    在闭包中声明好所有变量和方法,并通过一个 JSON Object 返回公有接口。
NameSpace.Hello = (function() {
    // 待返回的公有对象
    var self = {};
    // 私有变量或方法
    var name = 'world';
    // 公有方法或变量
    self.sayHello = function(_name) {
        return 'Hello' + (_name || name);
    }
    // 返回的公有对象
    return self;
}());

注意事项:

命名空间本身也是一个全局对象,在添加该“命名空间”时,有可能覆盖全局空间中的同名对象。因此我们需要在声明命名空间前进行检查,保证全局空间的安全。

var NameSpace = NameSpace || {};
  • 如果已存在同名的全局对象,那么就将所有的函数与变量都挂载到这同名的全局对象上。如果不存在,就创建一个新的全局对象。
  • 除了上述对全局对象是否存在进行判断之外,还能使用非破坏性处理命名空间的方式。
var YourGlobal = {
    namespace : function(ns) {
        var parts = ns.split("."),
            object = this,
            len = 0,
            i = 0;
        
        for(i = 0, len = parts.length; i < len; i++) {
            if(!object[parts[i]]) {
                object[parts[i]] = {};
            }
            object = object[parts[i]];
        }
        
        return object;
    }
};

/*
 * 同时创建 YourGlobal.Books 和 YourGlobal.Books.Event
 * 因为之前没有创建过它们,因此每个都是全新创建的
 */
YourGlobal.namespace("Books.Event");

// 现在你可以使用这个命名空间
YourGlobal.Books.Event.click = function() {
    //...
}

/*
 * 不会操作 YourGlobal.Books 本身,只给它添加 DOM 对象
 * 它会保持 YourGlobal.Books.Event 保持不变
 */
YourGlobal.namespace("Books.DOM");

// 仍然是合法的引用
console.log(YourGlobal.Books.Event.click);

// 同样可以在方法调用之后立即给它添加新属性
YourGlobal.namespace("Books").ANewBook = {};
  • 优点
    基于你的单全局对象使用 namespace() 方法可以让开发者放心地认为命名空间总是存在的。这样,每个文件都可以首先调用 namespace() 来声明开发者将要使用的命名空间,这样做不会对已有的命名空间造成任何破坏。这个方法可以让开发者从使用命名空间之前判断是否存在中解放出来。
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,907评论 6 506
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,987评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,298评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,586评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,633评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,488评论 1 302
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,275评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,176评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,619评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,819评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,932评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,655评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,265评论 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,871评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,994评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,095评论 3 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,884评论 2 354

推荐阅读更多精彩内容