局部变量陷阱

先看一段代码:

  1. function foo() {
  2. var a = "hello"
  3. b = "world"
  4. return a + b;
  5. }

这个函数执行完成之后返回helloworld,结果确实没有问题。但是里面有一个细节就是两个局部变量一个前边有var,另一个没有。这似乎并不影响执行,但是事实上没有var的话,这个变量就变成了全局变量。你在这个函数外面调用一下alert(b)试一下就知道了。一个莫名其妙冒出来的全局变量在有时候会引起匪夷所思的bug,所以一定要记得局部变量声明前一定要用var

变量声明提升

再看一段代码:

  1. var foo = "hello"
  2. function test() {
  3. alert(foo);
  4. var foo = "world";
  5. }
  6. test()

代码执行时,请问alert的内容是什么呢?答案是undefined。从直觉上来讲,运行alert的时候,foo这个变量没有声明,不存在。既然局部作用域没有,那就去全局作用域找,应该是"hello"才是啊。为什么呢?这就是变量声明提升。函数内的任何变量的声明都会被提升至函数的顶部,所以上面的test函数相当于

  1. function test() {
  2. var foo;
  3. alert(foo);
  4. foo = "world";
  5. }

所以会是undefined。这里注意提升的只是声明,赋值的代码还在原来的地方。并且声明的部分与声明的代码是否执行无关,比如

  1. function test() {
  2. alert(foo);
  3. if(false) {
  4. foo = "world";
  5. }
  6. }

把test函数修改成这个样子,执行结果仍然是一样。声明仍然被提前了,尽管这行代码将不会被运行。
这就是变量声明提升,我们再看一个例子

  1. function test() {
  2. foo1();
  3. foo2();
  4. function foo1(){alert("foo1")}
  5. var foo2 = function(){alert("foo2")}
  6. }

当test函数执行的时候,结果如何呢?结果是foo1被alert出来,然后JS报错"undefined is not a function"。这里在函数内部声明函数的提升有点区别。foo1,用函数的方式声明定义一个函数,则声明与函数体都被提升了,但是foo2用变量的方式定义函数,被提升的只有声明,函数体没有被提升,所以执行foo2()的时候,foo2没有被赋值,是undefined。

看起来由于JS里面有变量声明提升,那么就无法使用C语言里面的块级作用域,那么我确实想用,怎么办呢?有一个办法是用匿名函数,举例如下:

  1. var foo = "hello"
  2. function test() {
  3. alert(foo);
  4. (function(){
  5. var foo = "world";
  6. })()
  7. }
  8. test()

当test函数改成这个样子的时候,alert出来的结果就是"hello"了,匿名函数里面的foo就无法作用到其他地方。

闭包中的局部变量

我之前招聘前端,问了应聘者这样一个题目,说请写一个可以返回递增整数序列的函数,他给我的答案是这样

  1. var i = 0;
  2. function increase() {return i++;}

当反复调用函数increase的时候,就会返回递增序列。但是我接着问,如果有人失手改了i怎么样,他语塞。其实这个问题的关键是说要制造一个很安全的地方来放置这个i,让这个变量只能被一个函数访问,所以一个可选的答案是这样。此时increase就是会返回递增序列。并且i放在了一个很安全的地方。就不会被破坏了。

  1. var increase = (function() {
  2. var i = 0;
  3. return function(){return i++;}
  4. })();

JS的闭包是一个很值得讲的内容,好用,但是也有不少坑,下面我专门开一篇文章来讲。

全局变量

无数讲高质量代码的书籍都会提到全局变量这个大杀器,程序员们都被反复告诫,除非你又充足的理由,否则不要使用全局变量,JS也不例外。然而JS有一个特殊的地方就是没有main方法,入口的代码一定要写,但是写在这地方的代码难免要用到变量,这些变量就是全局变量了,那么如何处理这个问题呢?答案还是匿名函数,像这样:

  1. (function(){
  2. var a = "a";
  3. var b = "b";
  4. var c = "c";
  5. foo(a, b, c);
  6. })()

把入口代码放到一个匿名函数中,这样这些变量a, b, c就会被藏到这个函数里面,就避免了全局变量的出现。

JavaScript 进阶(二)变量作用域相关推荐

  1. JavaScript进阶(二)

    JavaScript进阶(二) 2019版黑马程序员javaScript进阶面向对象ES6 122集教程,哔哩哔哩链接:https://www.bilibili.com/video/BV1Kt411w ...

  2. python进阶15变量作用域LEGB

    原创博客链接:python进阶15变量作用域LEGB 作用域 "作用域"定义了Python在哪一个层次上查找某个"变量名"对应的对象.接下来的问题就是:&quo ...

  3. Javascript 中的变量作用域问题

    Js中的变量作用域问题: 1.没有块级作用域.Js中的变量作用域不是以{}为界的,不像C/C++/Java. 如: 1 if(true){ 2 var name = "qqyumidi&qu ...

  4. python类中变量作用域_python进阶14变量作用域LEGB

    作用域 "作用域"定义了Python在哪一个层次上查找某个"变量名"对应的对象.接下来的问题就是:"Python在查找'名称-对象'映射时,是按照什么 ...

  5. JavaScript解析顺序和变量作用域

    JavaScript基础之变量作用域. 一. 1.全局变量:全局变量的意思就是,在代码的不论什么地方都能够訪问到.注意:未定义 直接赋值的变量拥有全局属性. 2.局部变量:局部变量的意思就是,变量的作 ...

  6. 【JavaScript 笔记】— 函数高级(变量作用域、解构赋值、方法、高阶函数、闭包、箭头函数、generator)

    JavaScript个人笔记 变量作用域 变量提升 全局对象 windows 命名空间 局部作用域 常量 解构赋值 使用场景 方法 apply 装饰器 高阶函数(Array) map reduce m ...

  7. JavaScript进阶(四)

    JavaScript进阶(四) 2019版黑马程序员javaScript进阶面向对象ES6 122集教程,哔哩哔哩链接:https://www.bilibili.com/video/BV1Kt411w ...

  8. JavaScript进阶(三)

    JavaScript进阶(三) 2019版黑马程序员javaScript进阶面向对象ES6 122集教程,哔哩哔哩链接:https://www.bilibili.com/video/BV1Kt411w ...

  9. JavaScript进阶(一)

    JavaScript进阶(一) 2019版黑马程序员javaScript进阶面向对象ES6 122集教程,哔哩哔哩链接:https://www.bilibili.com/video/BV1Kt411w ...

  10. JavaScript入门(二)

    JavaScript入门(二) 哔哩哔哩链接:https://www.bilibili.com/video/BV1YW411T7GX?spm_id_from=333.999.0.0 七.代码讲解部分 ...

最新文章

  1. python主要运用于-python主要应用领域有哪些?看这一篇就够了
  2. ES6的新特性----Vue学习必备基础知识
  3. 几何视角看线性方程组解的情况
  4. SOAP 及其安全控制--转载
  5. 【SQL编程】Greenplum 数据库通过 timestamp 类型字段值实现数据的(日期时段筛选+时间时段筛选)跨天时段及不跨天时段SQL详情
  6. 超微服务器硬盘红灯_硬盘亮红灯服务器崩溃的解决方法
  7. 【287】◀▶ arcpy 常用类说明
  8. 美学评价_卡美学的真正美
  9. 用MATLAB实现神经网络
  10. 苹果系统c语言打开文件夹,MAC系统硬盘文件夹详解
  11. 7-60 致命的珠宝 (10分)
  12. vim - 格式化JSON
  13. 苹果审核技术支持URL导致的被拒解决方式
  14. 开放存取知识库及其数据采集规范的研究
  15. RAM汇编指令DMB、DSB、ISB、SEV等
  16. REST ful风格
  17. 支持网络和局域网共享文件的windows pe
  18. win 32 学习笔记(十) 对话框
  19. 搞不懂DMP是什么?看这里就够
  20. 理解字节序 [Understanding Big and Little Endian Byte Order]

热门文章

  1. mac上java文件如何编译_如何在Mac上用Java编译和运行程序?
  2. python实现矢量分级渲染_OpenLayer3实现分级渲染(初级版本)
  3. 2018-2019-1 《信息安全系统设计基础》 20165235 实验五 通信协议设计
  4. jedis远程连接linux报错
  5. bzoj 1753: [Usaco2005 qua]Who's in the Middle【排序】
  6. PHP高级——抽象类与接口的区别
  7. 将5:400等类型的数据转成哈希表
  8. Condition类必须配合Mutex使用,why?
  9. XML在线转化为JSON
  10. asp.net 中 SESSION和COOKIE的使用