开篇

当我们在开始学习任何一门语言的时候,都会接触到变量的概念,变量的出现其实是为了解决一个问题,为的是存储某些值,进而,存储某些值的目的是为了在之后对这个值进行访问或者修改,正是这种存储和访问变量的能力将状态给了程序。我们的程序中到处都充斥着对于状态的判断,根据不同的状态执行不同的逻辑。

我们试想一下,如果没有状态这个概念,程序虽然也能够执行一些简单的任务,但是它会受到很多的限制,所能完成的功能是有限制的,举个例子,没有状态你是如何执行循环语句?没有状态如何更加优雅地使用逻辑结构

仔细想想,好像是寸步难行,当然引入变量后帮我们解决了这个问题。

但是,引入变量和状态的概念之后会引起几个问题:这些变量住在哪里?换句话说,它们存储在哪里?最重要的是,程序需要它们的时候如何找到它们?

今天我们就一起学习一下这套存储和查找变量的规则,这套规则我们称之为:作用域。

作用域

我们来拆解一下这个词语,所谓的“”我们可以理解为:范围、区域,加上“作用”两个字所要表述的问题就是作用的范围、区域,比如国家的行政区域划分是为了便于管理,类比到程序源代码中作用域的出现也是为了便于对于变量做管理。

好,这里我们简单做一下总结:

  • 定义:作用域是指程序源代码中定义变量的区域。
  • 作用:作用域规定了如何查找变量,也就是确定当前执行代码对变量的访问权限。
  • 在javaScript中的应用 :JavaScript采用词法作用域(lexical scoping),也就是静态作用域

那什么又是 词法作用域或者静态作用域呢?

请继续往下看

静态作用域与动态作用域

因为javaScript采用的是词法作用域,函数的作用域在函数定义的时候就决定了。
而词法作用域相对的是动态作用域,函数的作用域是在函数调用的时候才决定的。
让我们看一个例子来理解词法作用域和动态作用域之间的区别:

var value = 1;function foo() {console.log(value);
}function bar() {var value = 2;foo();
}bar();
// 结果是 ???

上面的代码中:

  • 1.我们首先定义了一个value,并赋值为1;
  • 2.声明一个函数foo,函数的功能是打印 value 这个变量的值;
  • 3.声明一个函数bar,函数内部重新创建了一个变量 value 这个变量赋值为2;
    在函数内部执行了 foo() 这个函数;
  • 4.执行 bar() 这个函数

假设javaScript采用静态作用域,让我们分析下执行过程:

执行foo函数,首先从 foo 函数内部查找是否有变量 value ,如果没有
就根据书写的位置,查找上面一层的代码,我们发现value等于1,所以结果会打印 1。

假设javaScript采用动态作用域,让我们分析下执行过程:

执行foo函数,依然是从 foo 函数内部查找是否有局部变量 value。如果没有,
就从调用函数的作用域,也就是 bar 函数内部查找 value 变量,所以结果会打印 2。

上面在区分静态作用于和动态作用域的时候,我们已经说了如果是静态作用域,那么函数在书写定义的时候已经确定了,而动态作用域是函数执行过程中才确定的。

JavaScript采用的是静态作用域,所以这个例子的结果是 1。
我们在控制台中输入执行上面的函数,检验一下执行结果果然是 1。

动态作用域

那什么语言是采用的动态的作用域呢? 其实bash 就是动态作用域,
我们可以新建一个 scope.bash 文件将下列代码放进去,执行一下这个脚本文件:

#!/bin/bashvalue=1
function foo () {echo $value;
}
function bar () {local value=2;foo;
}
bar

上面代码运行的结果输出2很好解释,虽然在代码最上层定义了 value并赋值为1,但是在调用foo函数的时候,在查找 foo 内部没有 value 变量后,会在foo 函数执行的环境中继续查找,也就是在bar 函数中查找,很幸运我们找到了。

思考

最后,让我们看一个《JavaScript权威指南》中的例子:

// 例1:
var scope = "global scope";
function checkscope(){var scope = "local scope";function f(){return scope;}return f();
}
checkscope();// 例2:
var scope = "global scope";
function checkscope(){var scope = "local scope";function f(){return scope;}return f;
}
checkscope()();

让我们来分析一下上面例1的代码:

  • 1、定义一个变量 scope 并赋值 global scope;
  • 2、声明一个函数 checkscope ,在这个函数中 定义一个变量 scope 并赋值 local scope;
  • 3、在checkscope 函数中 又定义一个函数 f ,这个函数 只做了一件事:返回scope 这个变量;
  • 4、最后返回并执行 f 这个函数;
  • 5、调用checkscope

按照我们上面解释的javaScript中静态作用域理解,在执行 checkscope 这个函数的时候在函数内部执行的是f 这个函数,首先在 f 这个函数内部查找 scope 这个变量发现没有,继续在定义函数f的上面一层查找,发现在checkscope 这个函数作用域内 找到了scope的值 直接返回,至于 checkscope外面定义的scope没有理睬。

让我们来分析一下上面例2的代码:

  • 1、定义一个变量 scope 并赋值 global scope;
  • 2、声明一个函数 checkscope 在这个函数中 定义一个变量 scope 并赋值 local scope;
  • 3、在checkscope 函数中 又定义一个函数 f 这个函数 只做了一件事:返回scope 这个变量;
  • 4、最后单纯的返回 f 这个函数;
  • 5、调用checkscope

按照我们上面解释的javaScript中静态作用域理解,在执行 checkscope 这个函数的时候在函数内返回了函数f实际是在最外面调用的f但是由于javaScript是采用的词法作用域,因此函数的作用域基于函数创建的位置。

而引用《JavaScript权威指南》的回答就是:

JavaScript 函数的执行用到了作用域链,这个作用域链是在函数定义的时候创建的。嵌套的函数 f() 定义在这个作用域链里,其中的变量 scope 一定是局部变量,不管何时何地执行函数 f(),这种绑定在执行 f() 时依然有效。

但是在这里真正想让大家思考的是:

虽然两段代码执行的结果一样,但是两段代码究竟有哪些不同呢?

敬请期待下面一篇关于javaScript 中的执行上下文栈的相关内容。

参考:

  • 1、《你不知道的Javascript上卷》
  • 2、JavaScript深入之词法作用域和动态作用域

深入学习js之——词法作用域和动态作用域相关推荐

  1. 一文搞懂:词法作用域、动态作用域、回调函数、闭包

    把以前一直只限于知道,却不清晰理解的这几个概念完完整整地梳理了一番.内容参考自wiki页面,然后加上自己一些理解. 词法作用域和动态作用域 不管什么语言,我们总要学习作用域(或生命周期)的概念,比如常 ...

  2. JavaScript之词法作用域和动态作用域

    作用域 作用域是指程序源代码中定义变量的区域. 作用域规定了如何查找变量,也就是确定当前执行代码对变量的访问权限. JavaScript 采用词法作用域(lexical scoping),也就是静态作 ...

  3. JavaScript词法作用域和动态作用域

    2019独角兽企业重金招聘Python工程师标准>>> 作用域 作用域是指代码中定义变量的区域. 作用域规定了如何查找变量,也就是确定当前代码执行对变量的访问权限 JavaScrip ...

  4. 学习js的第五天【作用域,递归,对象,数组】

    学习js的第五天[作用域,递归,对象,数组] 一.作用域 ===>变量起作用的范围 1.局部作用域:在函数体中作用范围 2.全局作用域:在整个js全局起作用 var num=100 -----& ...

  5. 词法作用域和动态作用域

    JavaScript采用的是词法作用域 1.词法作用域 即函数定义时,即确定的作用域.js中的作用域链,在函数声明时候,就已经确定了,无论函数在何处调用,其作用域变量的查找都是按照定义是包含关系去查找 ...

  6. 静态作用域与动态作用域的区别(转)

    在计算机程序设计中,变量是名字(name)与实体(entity)的绑定(binding).一段程序代码中所用到的名字并不总是有效(或者说:可用的),而限定这个变量名字具有可用性的代码范围就是这个名字的 ...

  7. JS深入--词法作用域、执行上下文与闭包

    文章目录 词法作用域 执行上下文与词法环境 闭包 闭包练习 作用域链 REF   个人博客文章同步地址 词法作用域   JS 使用的是词法作用域(或称为静态作用域),函数的作用域在定义的时候就决定了, ...

  8. 【前端精进之路】JS篇:第4期 作用域

    文章目录 写在前面 作用域(Scope) 1.什么是作用域 2.全局作用域和函数作用域 3.块级作用域 作用域链 作用域链的创建 作用域与执行上下文 静态作用域与动态作用域 总结 写在前面 这里是小飞 ...

  9. 读书笔记-你不知道的JS上-词法作用域

    JS引擎 编译与执行 Javascript引擎会在词法分析和代码生成阶段对运行性能进行优化,包含对冗余元素进行优化(例如对语句在不影响结果的情况下进行重新组合). 对于Javascript来说,大部分 ...

最新文章

  1. Activiti 开始流程时存储发起人员USERID
  2. [FFmpeg] ffmpeg参数详解
  3. 049_汉字Unicode编码范围
  4. CentOS 查看系统 CPU 个数、核心数、线程数
  5. V-7 Openstack 在ceph中转换镜像格式
  6. 在SAP分析云里根据业务数据绘制词云(Word Cloud)
  7. payara 创建 集群_在Payara Server和GlassFish中配置密码
  8. sap获取系统时间_获取系统当前时间
  9. 美团与饿了么“鸡同鸭讲”后,又可以通过设置继续使用支付宝了?
  10. pass the URL parameters from the webseite page to Flash
  11. php怎么写书店系统,基于PHP技术的新华在线购书店系统
  12. 手机投屏电视机的原理
  13. 2021版Java同步器教程03:如何使用同步器 CyclicBarrier?
  14. egret引擎p2物理引擎(2) - 小球碰撞地面搞笑的物理现象
  15. 【阅读】《乔布斯的魔力演讲》
  16. 基于Java swing的多人,图片,语音聊天室
  17. 1m照片的宽和高是多少_1M照片是多大
  18. 京东秒杀系统是世界上最牛批的,不接受反驳!
  19. lambda表达式写法
  20. 从无序到有序的负熵力量

热门文章

  1. angularjs $state.go页面不刷新数据
  2. android Bundle的作用
  3. php 得到下周一和下周日
  4. 开源代码是下一轮攻击潮的重灾区
  5. Apache Cassandra 开源数据库软件修复高危RCE漏洞
  6. 施乐悄悄修复影响某些打印机中的严重缺陷
  7. vue 监听state 任意值变化、监听mutations actions
  8. 请求之前~HttpHandler实现媒体文件和图像文件的盗链
  9. (三)slatstack配置管理
  10. 009Maven_建立私服——报错问题