当前位置: 首页 > news >正文

JS 作用域和预解析

作用域

        通常来说,一段程序代码中所用到的名字并不总是有效和可用的,而限定这个名字的可用性的代码范围就是这个名字的作用域。作用域的使用提高了程序逻辑的局部性,增强了程序的可靠性,减少了名字冲突。

作用域分为全局作用域局部作用域

es6的时候新增了块级作用域{ }。

全局变量和局部变量

        根据作用域的不同,变量也可分为全局变量局部变量

        全局变量:在全局作用域下声明的变量,在全局都可以使用。

  • 在全局作用域下var声明的变量是全局变量
  • 如果在函数内部没有声明直接赋值的变量也属于全局变量(不建议使用)

        局部变量:在局部作用域下声明的变量(在函数内部定义的变量)只能在函数内部使用。

  • 在函数内部var声明的变量就是局部变量

二者区别

        全局变量:在任何一个地方都可以使用,只有在浏览器关闭时才会被销毁,因此比较占内存
        局部变量:只在函数内部使用,当其所在的代码块被执行时,会被初始化;当代码块运行结束后,就会被销毁,因此更节省内存空间。

作用域链

        内部函数访问外部函数的变量,采取的是链式查找的方式来决定取哪个值,这种结构我们称为作用域链 (就近原则)

下面这个代码,就近原则输出的是20。

预解析

js引擎分为两步:预解析 代码执行

(1)预解析  js引擎会把 js中所有的 var 还有function 提升到当前作用域的最前面

(2)代码执行  按照代码书写的顺序从上往下执行

预解析分为变量预解析(变量提升)函数预解析(函数提升)

变量预解析

变量提升 就是把所有的变量声明提升到当前的作用域最前面,不提升赋值操作。

console.log(num)  
var num=10;//相当于执行了以下代码var num;   //预解析,把变量声明提升上来
console.log(num);
num=10;  //赋值不提升

        上面这段代码预解析时会把变量声明提升到最前面,但不提升赋值操作,所以上面那个代码console.log(num)时num还没有被赋值,会输出undefined。

        注意函数表达式,是用一个变量存储函数的,预解析时只提升变量声明不提升变量赋值。所以函数表达式调用必须要写在函数表达式的下面,不然会出错。

// 函数表达式 调用必须要写在函数表达式的下面
fun();   //所以这里会输出undefined
var fun=function(){console.log(22);   
}//相当于执行了以下代码
var fun; //提升了变量声明
fun();
fun=function(){  //变量赋值不提升console.log(22);
}

函数预解析

函数提升 就是会把函数声明提升到当前作用域最前面,不提升调用操作。

fn();
function fn(){console.log(22)
}//预解析
//相当于执行下面的代码,所以不会报错
function fn(){   //函数声明console.log(22)
}
fn();   //函数调用

一道易错题:

f1();
console.log(c);
console.log(b);
console.log(a);
function f1(){var a=b=c=9;console.log(a);console.log(b);console.log(c);
}// 预解析
//相当于以下代码
function f1(){  //把函数声明提升var a;  //变量提升a=9;b=9;c=9;console.log(a);console.log(b);console.log(c);
}
f1();  //调用f1()输出9 9 9
console.log(c);
console.log(b);
console.log(a);//外面输出的是 9 9 undefined 

这道题易错点是:

        var a=b=c=9; 相当于var a=9;b=9;c=9;

        而不是集体声明,集体声明应该为 var a=9,b=9,c=9;

因为在函数内部没有声明直接赋值的变量也属于全局变量,所以b和c是全局变量。a是在函数内部声明的,所以a是局部变量。

http://www.lryc.cn/news/276975.html

相关文章:

  • 各种锁的概述
  • 【docker笔记】Docker容器数据卷
  • 大前端nestjs入门教程系列(四):如何nestjs整合mysql数据库
  • Android studio环境配置
  • 017、使用包、单元包及模块来管理日渐复杂的项目
  • Git提交规范详解
  • 线程与UI操作
  • ELK企业级日志系统分析系统
  • 11.23 校招 实习 内推 面经
  • Python战机
  • 外包做了5个月,技术退步一大半了。。。
  • 设计模式的艺术P1基础—2.2 类与类的UML图示
  • PCB 的正片、负片那些事儿
  • QT应用篇:QT解析与生成XML文件的四种方式
  • Android 正圆
  • C#,入门教程(13)——字符(char)及字符串(string)的基础知识
  • Tracert 与 Ping 程序设计与实现(2024)
  • 浅谈接口自动化测试
  • Hyperledger Fabric 核心概念与组件
  • 【C语言题解】 | 101. 对称二叉树
  • Baumer工业相机堡盟工业相机如何通过NEOAPI SDK实现相机掉线自动重连(C#)
  • [Vulnhub靶机] DriftingBlues: 5
  • 26 数字验证
  • echarts - xAxis.type设置time时该如何使用formatter的分级模板
  • 【代码随想录】刷题笔记Day47
  • 6.1 截图工具HyperSnap6简介
  • stable diffusion 人物高级提示词(二)衣物、身材
  • 外包做了1个月,技术退步一大半了。。。
  • docker-compose常用命令及.yaml配置模板
  • 工作随机:OEM(13.5)报错代理无法访问