函数作用域:函数内的变量不能在函数外调用
<script>function fn(){var num = 3;}console.log(num); // 显示未定义//es6以后,用var定义num不会形成块级作用域//块作用域{var num = 4;let n = 5;}console.log(num); //4console.log(n); //未定义</script>
不是块级作用域 - 全局
<script>for(var i=0;i<10;i++){}console.log(i) //10for(;i<20;i++){console.log(i) //[10到20]}</script>
使用let之后 - 变成了块级作用域
<script>for(var i=0;i<10;i++){}console.log(i) //未定义</script>
2、var let const 区别
全局作用域: {}外面的作用域,是全局作用域块级作用域: 只要{ }没有和函数结合在一起,就是块级作用域局部作用域(函数作用域): 函数后面的{}中的作用域
2-1 var 声明作用域
关键的问题在于,使用 var 操作符定义的变量会成为包含它的函数的局部变量
function test() {var message = "hi"; // 局部变量}test();console.log(message); // 出错// 在函数内定义变量时省 略 var 操作符,可以创建一个全局变量function test() {message = "hi"; // 全局变量}test();console.log(message); // "hi"
2-1-1 var 声明提升
使用这个关键字声明的变量会自动提升到函数作用域 顶部:function foo() {console.log(age);var age = 26;}foo(); // undefined 但不会报错
2-2 let 声明
let 跟 var 的作用差不多,但有着非常重要的区别。明显的区别是,let 声明的范围是块作用域, 而 var 声明的范围是函数作用域。
1. 暂时性死区
let 与 var 的另一个重要的区别,就是 let 声明的变量不会在作用域中被提升。
2. 全局声明
与 var 关键字不同,使用 let 在全局作用域中声明的变量不会成为 window 对象的属性(var 声明的变量则会)
3.条件声明
var 存在变量提升 let没有var 可以重复声明 let不可以
4. for 循环中的 let 声明
迭代变量的作用域仅限于 for 循环块内部:for (let i = 0; i < 5; ++i) {// 循环逻辑}console.log(i); // i 没有定义
2-3 const 声明
const 的行为与 let 基本相同,唯一一个重要的区别是用它声明变量时必须同时初始化变量且尝试修改 const 声明的变量会导致运行时错误
3、闭包
指子函数可以访问外部作用域变量的函数特性,即使在子函数作用域外也开可以访问;
如果没有闭包那么处理事件绑定,异步请求时都会变困难;
- js中所有的函数都是闭包
- 闭包一般在子函数本身作用域外执行,即延伸作用域
