Javascript学习笔记之函数篇(六) : 作用域与命名空间


Posted in Javascript onNovember 23, 2014

在之前的介绍中,我们已经知道 Javascript 没有块级作用,只有函数级作用域。

function test() { // a scope

    for(var i = 0; i < 10; i++) { // not a scope

        // count

    }

    console.log(i); // 10

}

Javascript 中也没有显示的命名空间,这就意味着一切都定义在全局作用域中。每一次引用一个变量时,Javascript 会往上遍历整个全局作用域直到找到该变量。如果遍历完整个全局作用域仍然没有找到该变量,则抛出一个 ReferenceError 错误。

请输入图片描述

隐式全局变量

// script A

foo = '42';

// script B

var foo = '42'

上面的两个例子产生不一样的效果。第一个将在全局作用域中定义变量 foo,而第二个则在当前作用域定义变量 foo。
我们一定要注意,如果不使用关键字 var 将会带来意想不到的影响。

// global scope

var foo = 42;

function test() {

    // local scope

    foo = 21;

}

test();

foo; // 21

由于在函数 test 内没用 var 来定义变量 foo,所以将覆盖函数外部的全局变量 foo。尽管看上去不是什么大问题,但是如果有成千上万行代码时,这将是个难以追踪的 bug。

// global scope

var items = [/* some list */];

for(var i = 0; i < 10; i++) {

    subLoop();

}

function subLoop() {

    // scope of subLoop

    for(i = 0; i < 10; i++) { // missing var statement

        // do amazing stuff!

    }

}

上例中,外部的循环将会在执行第一次的时候就停止,这是因为 subloop 函数内部的变量 i 将会覆盖外部的全局变量 i。我们只需要在函数内部加上一个 var 就可以避免这个错误,所以我们在定义变量时一定不要忘记加上关键字 var。除非我们确实希望对外部的全局变量造成影响。

局部变量

Javascript 中局部变量只可以通过两个方式产生,一是通过关键字 var 来声明,一是作为函数的形参。

// global scope

var foo = 1;

var bar = 2;

var i = 2;

function test(i) {

    // local scope of the function test

    i = 5;

    var foo = 3;

    bar = 4;

}

test(10);

此时,函数 test 内部的变量 i 和 foo 是局部变量,而 bar 则会覆盖外部的全局变量 bar。

提升(Hoisting)

Javascript 将会提升变量声明,这就意味着 var 表达式和函数声明都将被提升到作用域的顶部。

bar();

var bar = function() {};

var someValue = 42;

test();

function test(data) {

    if (false) {

        goo = 1;

    } else {

        var goo = 2;

    }

    for(var i = 0; i < 100; i++) {

        var e = data[i];

    }

}

上面的代码在运行之前, var 表达式和函数 test 的声明都将提升至顶部,因此程序将正常运行并不会报错。

// var statements got moved here

var bar, someValue; // default to 'undefined'

// the function declaration got moved up too

function test(data) {

    var goo, i, e; // missing block scope moves these here

    if (false) {

        goo = 1;

    } else {

        goo = 2;

    }

    for(i = 0; i < 100; i++) {

        e = data[i];

    }

}

bar(); // fails with a TypeError since bar is still 'undefined'

someValue = 42; // assignments are not affected by hoisting

bar = function() {};

test();

由于 Javascript 没有块级作用域,这不仅将提升 var 表达式,同时也会使得 if 结构变得不够直观。
在上例中,尽管看上去 if 在对全局变量 goo 进行操作,实际上,由于变量 goo 被提升,所以修改的是局部变量。
如果没有对提升规则有所了解,你可能会认为下面的代码将会抛出 ReferenceError 错误。

// check whether SomeImportantThing has been initialized

if (!SomeImportantThing) {

    var SomeImportantThing = {};

}

当然上面的代码是没有错误的,因为在代码在运行前,var 表达式已经被提升到顶部。

var SomeImportantThing;

// other code might initialize SomeImportantThing here, or not

// make sure it's there

if (!SomeImportantThing) {

    SomeImportantThing = {};

}

这里要推荐下 @nightire 凡哥的博文 《理解 JavaScript(二)》,里面对提升的讲解非常透彻。
名称解析顺序

当尝试在一个函数作用域内访问一个 foo 变量时,Javascript 将会按照下面的顺序查找:

当前作用域内是否有 var foo 的定义。
函数形参中是否有 foo 变量。
函数自身的名称是否为 foo。
跳到外层定义域,再从第一部开始查找起。
命名空间

一个最常见的问题就是命名冲突,这是因为 Javascript 只有一个全局作用域所带来的。但这个问题可以通过匿名的外部函数解决。

(function() {

    // a self contained "namespace"

    window.foo = function() {

        // an exposed closure

    };

})(); // execute the function immediately

上例中的匿名函数被认为是表达式,所以它们会被执行。

( // evaluate the function inside the parentheses

function() {}

) // and return the function object

() // call the result of the evaluation

当然我们也可以用其他方式来调用函数表达式,不同的结构,但是同样的效果。

// A few other styles for directly invoking the 

!function(){}()

+function(){}()

(function(){}());

// and so on...

总结

建议大家使用匿名的外部函数来将代码封装到空间内,这样不仅可以解决命名空间的冲突,同时也有利于程序的模块化。
此外,使用全局变量不是一个好习惯,这将带来高成本的维护代价而且容易产生错误。

命名空间同类型、函数、变量、模板等都属于实体(entity)。
实体的主要的共性是,可以具有名称。(此外,标签也可以具有名称,但它不是实体。)
而命名空间作用域是作用域中的一类统称,和块作用域、类作用域、函数原型作用域、函数作用域(仅对标签有效)并列。命名空间内声明的名称在命名空间作用域中。全局名称被认为在隐含的全局命名空间作用域中。

命名空间作用确实就是作用域,但是,他又不同于简单的作用域,你可以分多次在多处声明同一个命名空间,但是里面的内容不能重定义,他们最终都会合成一个命名空间,就像std,到处宏定义

Javascript 相关文章推荐
30个让人兴奋的视差滚动(Parallax Scrolling)效果网站
Mar 04 Javascript
Highcharts使用简例及异步动态读取数据
Dec 30 Javascript
JS中innerHTML和pasteHTML的区别实例分析
Jun 22 Javascript
Angular 4依赖注入学习教程之简介(一)
Jun 04 Javascript
Node.js学习教程之HTTP/2服务器推送【译】
Oct 31 Javascript
Vue-Router2.X多种路由实现方式总结
Feb 09 Javascript
Vue.js实现大转盘抽奖总结及实现思路
Oct 09 Javascript
jquery实现垂直手风琴导航栏
Feb 18 jQuery
jQuery三组基本动画与自定义动画操作实例总结
May 09 jQuery
vue axios请求成功却进入catch的原因分析
Sep 08 Javascript
Vue+element-ui添加自定义右键菜单的方法示例
Dec 08 Vue.js
浅谈react useEffect闭包的坑
Jun 08 Javascript
Javascript学习笔记之函数篇(五) : 构造函数
Nov 23 #Javascript
Javascript学习笔记之函数篇(四):arguments 对象
Nov 23 #Javascript
Javascript学习笔记之 函数篇(三) : 闭包和引用
Nov 23 #Javascript
js实例属性和原型属性示例详解
Nov 23 #Javascript
JS常用函数使用指南
Nov 23 #Javascript
浅谈JSON和JSONP区别及jQuery的ajax jsonp的使用
Nov 23 #Javascript
理解jQuery stop()方法
Nov 21 #Javascript
You might like
一篇有意思的技术文章php介绍篇
2010/10/26 PHP
thinkphp连贯操作实例分析
2014/11/22 PHP
thinkPHP实现MemCache分布式缓存功能
2016/03/23 PHP
Laravel中encrypt和decrypt的实现方法
2017/09/24 PHP
JS获取URL中参数值(QueryString)的4种方法分享
2014/04/12 Javascript
好好了解一下Cookie(强烈推荐)
2016/06/14 Javascript
Vue表单实例代码
2016/09/05 Javascript
jQuery文本框得到与失去焦点动态改变样式效果
2016/09/08 Javascript
javascript垃圾收集机制的原理分析
2016/12/08 Javascript
vue双向绑定的简单实现
2016/12/22 Javascript
利用原生JS与jQuery实现数字线性变化的动画
2017/02/24 Javascript
浅谈Node.js之异步流控制
2017/10/25 Javascript
AngularJS基于http请求实现下载php生成的excel文件功能示例
2018/01/23 Javascript
AjaxUpLoad.js实现文件上传
2018/03/05 Javascript
Vue进度条progressbar组件功能
2018/04/17 Javascript
基于JS实现前端压缩上传图片的实例代码
2019/05/14 Javascript
JS简单数组排序操作示例【sort方法】
2019/05/17 Javascript
js实现图片粘贴到网页
2019/12/06 Javascript
pyramid配置session的方法教程
2013/11/27 Python
Python实现的中国剩余定理算法示例
2017/08/05 Python
pyqt5实现俄罗斯方块游戏
2019/01/11 Python
python操作kafka实践的示例代码
2019/06/19 Python
在python中创建指定大小的多维数组方式
2019/11/28 Python
python 通过视频url获取视频的宽高方式
2019/12/10 Python
vue常用指令代码实例总结
2020/03/16 Python
python爬虫beautifulsoup库使用操作教程全解(python爬虫基础入门)
2021/02/19 Python
css3实现背景模糊的三种方式(小结)
2020/05/15 HTML / CSS
美国在线乐器和设备商店:Musician’s Friend
2018/07/06 全球购物
请写出char *p与"零值"比较的if语句
2014/09/24 面试题
董事长秘书岗位职责
2013/11/29 职场文书
电厂厂长岗位职责
2014/01/02 职场文书
高中化学教学反思
2014/01/13 职场文书
五年级小学生评语
2014/12/26 职场文书
详解CocosCreator消息分发机制
2021/04/16 Javascript
mysql对于模糊查询like的一些汇总
2021/05/09 MySQL
opencv深入浅出了解机器学习和深度学习
2022/03/17 Python