深入了解JavaScript中var和let的区别

简介: 深入了解JavaScript中var和let的区别

变量


ECMAScript变量是松散类型,意思就是说变量可以保存任何类型的数据,每个变量只不过是用于保存任意值的命名占位符。在JavaScript中,有3个关键字可以声明变量:var、constlet,其中varECMAScript的所有版本中都可以使用,而constlet只能在ECMAScript 6及更晚的版本中使用

var 关键字

如果我们想定义变量,我们可以使用var操作符(这里要注意var也是一个关键字~),后跟变量名

var message;

这行代码会定义一个名为message的变量,用于保存任何类型的值如果在不初始化的情况下,默认为undefinedECMAScript实现变量初始化,因此可以同时定义变量并且设置它的值

var message = "hi";

通过上方代码我们可以了解到,message被定义为一个保存字符串值hi的变量,像这样初始化变量不会将它标识为字符串类型,这仅仅是一个简单的赋值而已,我们不仅可以改变保存的值,同时还可以改变值的类型

var message = "hi"
message = 100 ;

通过上方代码我们可以看到,message首先被定义为一个用于保存字符串值hi的变量,然后又被重写为保存了数值100,虽然不推荐改变变量保存值的类型,但是这个在ECMAScript里面是非常有效的

var声明作用域

我们在定义一个函数变量的时候,通常都会考虑他们的作用域,使用var操作符定义的变量将会成为包含它的函数的局部变量,比如用var在一个函数内部定义一个变量,就意味着一旦出了这个区域,此定义无效

<script>
function test() {
    var message = "hi";
}
test();
console.log(message);
</script>

从上方代码我们可以看到,message变量是在函数内部使用var定义的,函数名叫test() ,调用它会创建这个变量并且会给它赋值,调用结束之后这个变量就会被销毁,因此上方代码的倒数第二行会报语法错误,如果我们想要解决这个问题,把message定义成全局变量即可

<script>
function test() {
    message = "hi";
}
test();
console.log(message);
</script>

去掉之前的var操作符之后,message变成了全局变量,我们只需要调用一次函数test() ,就可以定义这个变量,并且在函数外部访问

注意注意!!!

我们虽然可以通过省略var操作符来定义全局变量,但是不推荐这么做,在局部作用域中定义的全局变量很难去维护,也会给我们造成一定的困惑,因为我们不能确定这个var是不是有意而为之,如果像这样给未声明的变量赋值,则会导致抛出ReferenceError

如果我们需要定义多个变量,我们可以在一条语句中用逗号分隔每个变量(及可选的初始化)

<script>
    var message = "hi";
    found = false;
    age = 29;
</script>

这里我们定义了3个初始化变量,因为ECMAScript松散类型的,所以使用不同数据类型初始化的变量可以用一条语句来声明,插入换行和空格缩进不是必需品(因为这里不是Python~)

小tip

严格模式下,不能定义名为evalarguments的变量,否则会导致语法错误

var声明提升

我们在使用var的时候,下面的代码不会报错,因为使用这个关键字声明的变量会自动提升到函数作用域的顶部

<script>
    function foo() {
        console.log(age);
        var age = 26;
    }
    foo();
</script>

之所以这个代码不会报错,是因为ECMAScript已经帮我们解决啦~(它自己会自动调整阅读顺序),这就是所谓的提升,也就数把所有变量声明都拉到函数作用域的顶部,此外,反复多次使用var声明同一个变量也是没有问题的

let声明

letvar的作用差不多,但是有着十分重要的区别,最明显的区别就是,let声明的范围是块作用域,而var声明的范围是函数作用域

<script>
   if(true){
       var name = "Matt"
       console.log(name);  //Matt
   }
   console.log(name);    //Matt
   if(true){
       let age = 26 ;
       console.log(age)   //26
   }
   console.log(age)        //ReferenceError
</script>

在这里,age变量之所以不能在if块外部被引用,是因为它的作用域仅限于该块内部,块作用域是函数作用于的子集,因此适用于var的作用域也适用于let

let不允许同一个块作用域中出现冗余声明,这也会报错

var name ;
var name ;
let age ;
let age ;    //SyntaxError  标识符age已经声明过了

当然,JavaScript引擎会记录用于变量声明的标识符及其所在的块作用域,因此嵌套相同的标识符不会报错, 这是因为在同一个块中没有重复声明

<script>
    var name = "Nicholas";
    console.log(name);        // Nicholas
    if (true) {
        var name = "Matt";
        console.log(name);    // Matt
    }
    let age = 30;
    console.log(age);         // 30
    if (true) {
        let age = 26;
        console.log(age);     // 26
    }
</script>

对声明冗余报错不会因混用letvar而受影响,这两个关键字声明的并不是不同类型的变量,它们只是指出变量在相关作用域如何存在

var name ;
let name ;    //SyntaxError
let age ;
var age ;     //SynatxError


相关文章
|
1月前
|
前端开发 JavaScript UED
JavaScript防抖和节流的使用及区别
JavaScript防抖和节流的使用及区别
101 57
|
4天前
|
存储 JavaScript 前端开发
JavaScript 数据类型详解:基本类型与引用类型的区别及其检测方法
JavaScript 数据类型分为基本数据类型和引用数据类型。基本数据类型(如 string、number 等)具有不可变性,按值访问,存储在栈内存中。引用数据类型(如 Object、Array 等)存储在堆内存中,按引用访问,值是可变的。本文深入探讨了这两种数据类型的特性、存储方式、以及检测数据类型的两种常用方法——typeof 和 instanceof,帮助开发者更好地理解 JavaScript 内存模型和类型检测机制。
10 0
JavaScript 数据类型详解:基本类型与引用类型的区别及其检测方法
|
10天前
|
存储 JavaScript 前端开发
js中函数、方法、对象的区别
js中函数、方法、对象的区别
11 2
|
21天前
|
JavaScript 前端开发
【JavaScript】let,const和var的区别
总的来说,随着ECMAScript 6(ES6)及后续版本的推广,`let`和 `const`因其增强的块级作用域和对变量行为的更严格控制,逐渐成为现代JavaScript编码实践中推荐使用的变量声明方式。而 `var`由于其历史遗留的局限性,正逐渐被边缘化,但在维护老代码或处理特定兼容性需求时仍需了解。
19 3
|
1月前
|
前端开发 JavaScript 开发者
React 和 Vue.js 框架的区别是什么?
React 和 Vue.js 框架的区别是什么?
|
1月前
|
JavaScript 前端开发
JavaScript中单引号和双引号的效果的一点区别
JavaScript中单引号和双引号的效果的一点区别
|
1月前
|
JavaScript 前端开发
JavaScript let 和 const
JavaScript let 和 const
13 3
|
1月前
|
前端开发 JavaScript API
JavaScript 的宏任务和微任务有什么区别
【9月更文挑战第6天】JavaScript 的宏任务和微任务有什么区别
37 4
|
1月前
|
JavaScript 前端开发
JS基础 if和switch区别
本文比较了JavaScript中的`if`语句和`switch`语句的区别,并通过示例代码展示了它们的用法,解释了何时使用`if`更合适以及`switch`语句的特点和应用场景。
17 0
|
JavaScript 前端开发 PHP
JavaScript中var关键字的使用详解
这篇文章主要介绍了JavaScript中var关键字的使用,是JavaScript入门学习中的基础知识
143 0