一、作用域(what?)

  官方解释是:“一段程序代码中所用到的名字并不总是有效/可用的,而限定这个名字的可用性的代码范围就是这个名字的作用域。”
  单从文字理解比较难懂,举个栗子:

  1. function outer(){
  2. // 声明变量
  3. var name = "ukerxi";
  4.  
  5. // 定义内部函数
  6. function inner() {
  7. console.log(name); // 可以访问到 name 变量
  8. }
  9. }
  10. console.log(name); // 报错,undefined

  其中变量name声明在 oute r函数中,当在 outer 中定义一个 inner 函数进行输出 name,可以得到正确的值,而在 outer 外进行输出 name 出现 undefined 错误;在此可以看出 outer 函数即为 name 变量的作用域(证明过程比较粗略,但结论还是正确的--_--);

二、作用域(why?)

  作用域的使用,提高了程序的逻辑的局部性,增强程序的可靠性,以及避免命名冲突;为代码的模块化开发提供便利;根据上面提到的函数作用域,name 变量被局限在了 outer 函数中,在其他的函数中也可以定义相同名字的变量,两者之间不会互相影响;

三、js 中的作用域

  先说ES5版本及更低版本的,因为在 ES6 上,重新定义了几个决定作用域的关键字;

  1. 没有块级作用域
    在javaScript中,不像C、java等拥有块级作用域;常见块级作用域,例如:

    1. // C语言实现
    2. for(int i= 0; i<10; i++){
    3. // 中括号里面就是块级作用域
    4. }
    5.  
    6. if(ture){
    7. int i = 1;
    8. // 这里也是块级作用域
    9. }
    10. printf("%d/n",i); // --“use an undefined variable:i”
    11. //这里是访问不到for语句中的i与if语句中的i变量的

    当然这些在javaScript中是没有的,一般来说只有块级作用域;所以使用时须注意作用域的影响,例如:

    1. for(var i= 0; i<10; i++){
    2. // do something
    3. }
    4.  
    5. console.log(i); // --10

    在程序设计过程中可以使用函数作用域,进行模拟块级作用域;例如:

    1. function loop(){
    2. for(var i= 0; i<10; i++){
    3. // do something
    4. }
    5. }
    6. loop();
    7. console.log(i); // --undefined

    javaScript是灵活可变的,同样上面这个例子,可以使用自执行函数重写实现,这样就减少了调用这一步;

    1. (function (){
    2. for(var i= 0; i<10; i++){
    3. // do something
    4. }
    5. }());
    6. console.log(i); // --undefined
  2. ES6中的作用域
    在ES6中新增加了(let,const)关键字,进行定义变量,解决了没有块级作用域的限制;
    let:let允许你声明一个作用域被限制在块级中的变量、语句或者表达式。与var关键字不同的是,它声明的变量只能是全局或者整个函数块的。let声明的变量只在其声明的块或子块中可用,这一点,与var相似。二者之间最主要的区别在于var声明的变量的作用域是整个封闭函数。
    具体如下:
    1. function range () {
    2. // let 和var 相同的地方,都有函数作用域
    3. var name = 'ukerxi';
    4. let nameOuter = 'outer';
    5.  
    6. for (var j = 0; j < 1; j++) {
    7. console.log(name)
    8. }
    9. console.log("输出j变量", j); // ==> 1
    10. for (let i = 0; i < 1; i++) {
    11. console.log(nameOuter)
    12. }
    13. console.log("输出i变量", i); // 报错 undefined
    14. }

    可以看出,使用let 定义的i变量,在for语句外进行输出时,会进行报错,说明i不在该作用域内,i的作用域在for包裹的作用内;

  3. 作用域链
    每个函数都有自己的执行环境,包含当前环境的变量访问关系,与之相关联的就是“变量对象”,如果是当前函数的变量对象,也可称为“活动对象”;此对象中包含了,当前函数可访问的变量及函数;变量对象,最开始包含的对象是参数的arguments对象,然后是在函数中定义的其他变量及方法;例如:
    1. function fn(name){
    2. var text = "test";
    3. }
    4. // 变量对象中包含:命名函数fn变量、参数name、内部变量test

    当然这个变量对象是不可访问的,只提供后台引擎编译执行使用;当定义有多个变量对象嵌套,这些变量对象就组成了作用域链;例如:

    1. var name = "global";
    2. function super() {
    3. var name = "super";
    4. function sub(){
    5. var name = "sub";
    6. }
    7. }

    作用域链:

    在作用域最前端的是活动对象,而最后端是全局执行环境window(浏览器宿主中);变量访问原则是,根据作用域前端往上进行搜索,如果提前搜索到变量,则停止搜索,例如上面这个例子中,name变量的值是"sub"因为其在最前端的变量对象中已经定义了,就不会往上继续检索;

  4. 延长作用域
    有两种方法可以将作用域进行延长:
    ①、try-catch 语句的catch块
    ②、with 语句

    两个语句都是在原本的作用域最前端进行添加一个变量对象;例如:

    1. var name = "global";
    2. function test(){
    3. var name = "sub";
    4. with(window){
    5. console.log(name);
    6. }
    7. }
    8.  
    9. test(); // -- "global"

    作用域链:


    所以检索变量时,会先在最前端的window变量对象中检索;当然,在严格模式下已经禁用了with语句,编程时,最好向后兼容,废弃使用with语句;

  5. 执行环境只与函数的声明及定义位置有关
    当一个函数定义后其执行环境与作用域链就已经确定了,不会因为执行位置改变而改变,具体例子:
    1. var name = "global";
    2. function getName(){
    3. console.log(name);
    4. }
    5.  
    6. function test (){
    7. var name = "inner";
    8. getName();
    9. }
    10.  
    11. // 执行test
    12. test(); // -- global

    运行test 函数,其中test 函数执行的是 getName 进行输出 name 变量,输出的是全局变量的信息;即当 getName 定义时就已经确定了自己的作用域及执行环境,因而不会因为执行位置的不同而输出不同的信息;当然有一种情况不一样,那就是灵活的 this

  6. this的动态绑定
    与作用域链及执行环境不同,this是根据执行时的接受者进行绑定的,改变this的几种方法:
    ①、new 关键字
    ②、call / apply 方法
    ③、直接调用构造函数

    具体例子如下:

    1. // 声明一个类
    2. function Person (){
    3. this.name = "ukerxi";
    4. }
    5. // 使用new关键字,使this执行新建对象
    6. // 其实是构造函数默认返回this
    7. var men1 = new Person(); // this绑定到men1上
    8.  
    9. // 声明一个空对象,使用call/apply 进行绑定
    10. var men2 = {};
    11. Person.call(men2); // this绑定到men2上
    12.  
    13. // 直接执行构造函数
    14. Person(); // this绑定到window上(使用严格模式则会报错,this指向undefined)

【结束语】

系列文章,包括了原创,翻译,转载等各类型的文章;一方面是为了自己总结,另一方面页希望可以共享知识;在技术方面有输入,也要有所输出,才能更进一步!文章基于自己的实践、阅读及理解,如有不合理及错误的地方,烦请各大佬评论指出,以便改正,感谢!

js作用域与执行环境(前端基础系列)的更多相关文章

  1. JS 作用域(执行环境)与作用链---JS 学习笔记(二)

    一  作用域(执行环境) 作用域:定义了变量和函数有权访问的其他数据,决定了他们各自的行为.--------<JS高级程序设计>4.2 好难理解啊~参考了参考尤克希的博客内容,大体上理解了 ...

  2. js中的执行环境及作用域

    最近在面试时被问到了对作用域链的理解,感觉当时回答的不是很好,今天就来说说js中的作用域链吧. 首先来说说js中的执行环境,所谓执行环境(有时也称环境)它是JavaScript中最为重要的一个概念.执 ...

  3. js中的执行环境和作用域链

    首先介绍一些即将用到的概念: 执行环境:  它定义了变量和函数有权访问其他数据的范围,每一个执行环境都有一个与之关联的变量对象,环境中定义的所有变量和函数都保存在这个变量对象中.   所有javasc ...

  4. JS中的执行环境和作用域

    window 是最大最外围的执行环境,然后每个函数都有自己的执行环境.JS代码是从上到下执行的,单纯的用语言描述可能会有点绕,而且不大直观.我们看着代码来 console.log('global be ...

  5. “use strict” 严格模式使用(前端基础系列)

    ECMAscript5添加一种严格模式的运行模式("use strict"),让你的js语句在更加严格的环境下进行运行: 一.主要作用: 消除版本javascript中一些不合理及 ...

  6. 深入理解js——作用域和上下文环境

    如图除全局作用域外,每个函数都会创建自己的作用域.作用域在函数定义时就确定了,而不是在函数调用时确定. 下面按照程序执行的步骤加上上下文环境. 第一步:程序加载时已经确定全局上下文环境,并随着程序的执 ...

  7. navigator.userAgent浏览器检测(前端基础系列)

    对于前端来说,浏览器检测已经不陌生了,在做一些页面是,需要针对不同的浏览器进行处理不同的逻辑,最简单的就是区分pc和移动端的浏览器,或是android 和ios下的浏览器. 一.浏览器检测的由来?  ...

  8. 前端基础系列——CSS规范(文章内容为转载)

    原作者信息 作者:词晖 链接:http://www.zhihu.com/question/19586885/answer/48933504 来源:知乎 著作权归原作者所有,转载请联系原作者获得授权. ...

  9. javascript 命名空间与运用(前端基础系列)

    所谓代码,当你随便命名一个变量:var name = "ukerxi"; 就是一句代码:但当你的代码写出来后,对于后续维护及阅读的人,就可以看出代码是否,易读,易理解:优雅的代码总 ...

随机推荐

  1. C#第二篇——关于C#中的正则表达式

    在C#中,正则表达式是用来进行查询在给出的一串字符中的某些字符或者数字的工具.与在办公软件中的查找功能相似,可以用精确查找也可以用模糊查找. 元字符: 元字符 说明 . 匹配除换行符以外的任意字符 \ ...

  2. Problem N

    Problem Description The cows have purchased a yogurt factory that makes world-famous Yucky Yogurt. O ...

  3. DOM 遍历-同胞

    在 DOM 树中水平遍历 有许多有用的方法让我们在 DOM 树进行水平遍历: siblings() next() nextAll() nextUntil() prev() prevAll() prev ...

  4. <template> 标签

    <template> 元素,用于描述一个标准的以 DOM 为基础的方案来实现客户端模板.该模板允许你定义一段可以被转为 HTML 的标记,在页面加载时不生效,但可以在后续进行动态实例化.( ...

  5. display: run-in

    If a sibling block box (that does not float and is not absolutely positioned) follows the run-in box ...

  6. oracle 归档模式开启后数据库宕机解决过程

    首先按照网友说的shutdown immediately,结果hang了半个小时也么反应. 然后检查日志,全盘搜索.trc,发现 (D:\app\oracle\diag\rdbms\cms1u\cms ...

  7. 第一章 Linux系统介绍与环境搭建准备

    1.操作系统: Operating System,简称OS,它是应用程序运行以及用户操作必备的基础环境支撑,是计算机系统的核心. 操作系统就是处于用户与计算机系统硬件之间用于传递信息的系统程序软件. ...

  8. 入门到熟练-Eclipse开发工具

    1. 概述 本文用于Eclipse说明开发功能的各种配置.希望可以帮助到对于Eclipse工具设置不同熟练的朋友,快速上手Eclipse开发工具. 2. Eclipse的配置 2.1. 设置Eclip ...

  9. C#生成缩略图源码

    先看调用的方法: ).ToUpper())                {                    case "JPG":                      ...

  10. 初始MyBatis

    初始MyBatis 框架的概念: 框架是一个提供可重复的功用结构的半成品.它为我们构建新的应用程序提供了极大的便利,一方面提供了可以拿来就用的工具,更重要的是提供了可重用的设计.D 框架技术的优势: ...