js中变量提升和函数提升
变量提升和函数提升的总结
我们在学习JavaScript时,会遇到变量提升和函数提升的问题,为了理清这个问题,现做总结如下,希望对初学者能有所帮助
我们都知道 var 声明的变量有变量提升,而 let 声明的变量和 const 声明的常量没有变量提升,但是为什么 let 声明的变量就不会变量提升呢?
首先我们得理解变量和函数的创建过程:
变量声明 ==>变量初始化 ==>变量赋值
函数声明 ==> 初始化 ==> 赋值
而变量提升和函数提升要完成的工作为:
变量声明 ==> 变量初识化(undefined)
函数声明 ==> 初始化 ==> 赋值(仅仅是赋值,并没有运行函数体)
暂时性死区的概念:
ES6规定,如果代码区块中存在 let 和 const 命令声明的变量,这个区块对这些变量从一开始就形成了封闭作用域,直到声明语句完成,这些变量才能被访问(获取或设置),否则就会报错 ReferenceError。这在语法上称为“暂时性死区”,即代码块开始到变量声明语句完成之间的区域。通过 var 声明的变量拥有变量提升,没有暂时性死区。
因为有暂时性死区的存在,所以 let 声明的变量和 const 声明的常量才 不能 变量提升
从变量创建过程来理解:
let 声明被提升了,但是并没有做初始化工作
const 声明被提升,也是没有做初始化工作(const只有声明和初始化过程,没有赋值过程)
let 和 const 都没有提升所要满足的工作,所以就无法提升
var 的声明被提升,初始化被提升
function 的声明、初始化、和赋值都被提升
var 声明的变量提升(声明、初始化都被提升,但是赋值没有提升)
实例
console.log(a) // undefined
var a = 'hello world'
console.log(a) // 'hello world'
实际执行过程如下
var a
console.log(a) // undefined
a = 'hello world'
console.log(a) // 'hello world'
function 声明的函数提升(声明、初始化、赋值都提升了)
实例
fn() // 'hello world'
function fn() { console.log('hello world') }
fn() // 'hello world'
实际执行过程如下
function fn() { console.log('hello world') } // 注意:此时函数体并没有运行
fn() // 'hello world'
fn() // 'hello world'
既有 var 声明的变量提升,又有 function 声明的函数提升
实例
变量名和函数名不同名
console.log(a) // undefined
fn()
var a = 100
console.log(a) // 100
function fn() {
console.log(a) // undefined
var a = 10
console.log(a) // 10
}
var a = 200
console.log(a) // 200
实际执行过程如下
function fn() {
console.log(a)
var a = 10
console.log(a)
}
var a (全局变量)
var a (局部变量)
console.log(a) // undefined
fn() // undefined 10
a = 100
console.log(a) // 100
a = 200
console.log(a) // 200
变量名和函数名同名
情况一:变量由 函数类型 变为 基本数据类型(如 Number)
var a = 100
console.log(a) // 100
a() // 报错(TypeError:a is not a function)
function a() {
console.log('hello world')
}
实际执行过程如下
function a() {
console.log('hello world')
}
var a // 用 var 重新声明了变量 a,但此时变量 a 指向的还是函数体本身,值未改变
a = 100 // 重新给 a 赋值为 100,a 的值发生了改变,由函数类型变为了number类型
console.log(a) // 100
a() // 报错:此时 a 已经不是函数 a 了
情况二:变量重新赋值,但未立即改变变量的值
console.log(a) // a(){ console.log('hello world')} 即 a 函数本身
a() // 'hello world'
var a = 100
var a = 200
var a = 300
console.log(a) // 300
function a() {
console.log('hello world')
}
实际执行过程如下
function a() {
console.log('hello world')
}
var a // 此时虽然重新声明了 a 变量,但是并没有改变 a 的值,此时的 a 还是个函数
console.log(a) // a(){ console.log('hello world')} 即 a 函数本身
a = 100 // 重新给 a 变量赋值 100
a = 200 // 重新给 a 变量赋值 200
a = 300 // 重新给 a 变量赋值 300
console.log(a) // 100(此时变量 a 已经由 函数类型变为 基本数据类型number,且值为 100)
a() // 报错:此时 a 已经不是函数 a 了
用 let 定义的变量和用 const 定义的常量 没有提升
用 let 定义的变量
在变量定义之前使用改变量 会报错
console.log(a) // 报错 Cannot access 'a' before initialization
let a = 10
在变量定义之后使用改变量 才能正常运行
let a = 10
console.log(a) // 10
总结:
1. 用 var 定义的变量有变量提升,用 function 定义的函数有函数提升
2. 函数提升优先级高于变量提升,即优先执行函数提升
3. let 定义的变量和 const 定义的常量不会提升
4. 重复用 var 定义的同一个变量是合法的,如果重复的声明有初始值,那就赋值;如果没有初始值,那么它不会对变量原来保存的值不会产生影响
5. 不能用 let 重复定义同一个变量,会造成错误
js中变量提升和函数提升的更多相关文章
- JS中变量名和函数名重名
今天骚凯问了一道变量名冲突的题目,感觉很有意思,顺便也复习一下预解析的一些知识,有不对的地方忘前辈大神指正,题目是这样的: var a=100; function a(){ console.log(a ...
- js中的变量提升和函数提升
从上周开始,我所在的学习小组正式开始了angular的学习,angular是全面支持es6的,所以语法上和以前的angular有了很大的不同,比如变量声明时就抛弃了var,而选择了let和const: ...
- 关于JS中变量提升的规则和原理的一点理解
关于变量提升,以前在一些教程和书籍上都听到过,平时开发中也知道有这个规律,但是今天突然在一个公开课中听到时,第一反应时一脸懵逼,然后一百度,瞬间觉得好熟悉啊,差点被这个概念给唬住了,不信我给你 ...
- JS 变量提升与函数提升
JS 变量提升与函数提升 JS变量提升 变量提升是指:使用var声明变量时,JS会将变量提升到所处作用域的顶部.举个简单的例子: 示例1 console.log(foo); // undefined ...
- js变量提升与函数提升的详细过程
大家好,这里是「 从零开始学 Web 系列教程 」,并在下列地址同步更新...... github:https://github.com/Daotin/Web 微信公众号:Web前端之巅 博客园:ht ...
- JS——变量提升和函数提升
一.引入 在了解这个知识点之前,我们先来看看下面的代码,控制台都会输出什么 var foo = 1; function bar() { if (!foo) { var foo = 10; } aler ...
- 关于JS中变量提升的规则和原理的一点理解(二)
上篇文章中讲到变量提升和函数提升的先后顺序时蒙了,后来去查了一下资料,特别整理一下. 在<你不知道的JavaScript(上卷)>一书的第40页中写到:函数会首先被提升,然后才是变量. 书 ...
- 谈谈javascript中的变量提升还有函数提升
在很多面试题中,经常会看到关于变量提升,还有函数提升的题目,所以我就写一篇自己理解之后的随笔,方便之后的查阅和复习. 首先举个例子 foo();//undefined function foo(){ ...
- JS逻辑题 技术点: 1). 变量提升 2). 函数提升 3). 预处理 4). 调用顺序
考查的技术点: 1). 变量提升 2). 函数提升 3). 预处理 4). 调用顺序 var c = 1; function c(c) { console.log(c); var c = 3; ...
随机推荐
- SQL 查询语句总是先执行 SELECT?你们都错了
很多 SQL 查询都是以 SELECT 开始的.不过,最近我跟别人解释什么是窗口函数,我在网上搜索"是否可以对窗口函数返回的结果进行过滤"这个问题,得出的结论是"窗口函数 ...
- HAOI2008 木棍分割 数据结构优化dp+二分答案
很久之前打的题,现在补篇博客 打滚动数组 #E. 木棍分割 Accepted 100 1712 ms 1512 KiB 2019-05-07 17:01:23 Short 不打滚动数组 #419. ...
- WPF Frame 的 DataContext 不能被 Page 继承
转载至https://blog.csdn.net/sinat_31608641/article/details/88914517 已测试解决方案可行,因为WPF相关资料稀少,防止日后404,特搬运到自 ...
- [源码解析] 深度学习分布式训练框架 horovod (7) --- DistributedOptimizer
[源码解析] 深度学习分布式训练框架 horovod (7) --- DistributedOptimizer 目录 [源码解析] 深度学习分布式训练框架 horovod (7) --- Distri ...
- VsCode中代码折叠快捷键
ctrl+K ctrl+[ 折叠本级 ctrl+K ctrl+] 取消折叠本级 ctrl+K ctrl+0 折叠全部 ctrl+K ctrl+J 取消折叠全部
- 服务器通信REST、gRPC,Swagger/OpenAPI,Consul
服务间的通信方式是在采用微服务架构时需要做出一个最基本的决策.默认的选项是通过 HTTP 发送 JSON,也就是所谓的 REST API.我们也是从 REST 开始的,但最近我们决定改用 gRPC. ...
- 图解协程调度模型-GMP模型
现在无论是客户端.服务端或web开发都会涉及到多线程的概念.那么大家也知道,线程是操作系统能够进行运算调度的最小单位,同一个进程中的多个线程都共享这个进程的全部系统资源. 线程 三个基本概念 内核线程 ...
- rsync 基本使用
基本参数 # rsync -P test.tar.gz ./ test.tar.gz 395,706,368 48% 377.34MB/s 0:00:01 Or # rsync -avPh test. ...
- C# 实现复制Excel内容到DataGridview中
业务要求:复制:将Excel内容复制到datagridview中 最终效果:复制Excel内容,点击datagridview中的某个单元格,顺着这个单元格自动填充自动增加行.偷懒了,没写填充在选择哪些 ...
- Linux:Ubuntu银河麒麟防火墙操作
查看防火墙状态 #防火墙状态 sudo ufw status inactive状态是防火墙 关闭 状态 active状态是防火墙 开启 状态 开启防火墙 #开启防火墙 sudo ufw enable ...