紫影基地

 找回密码
 立即注册
查看: 132|回复: 0

Javascript 中 var 和 let 、const 的区别

[复制链接]
阅读字号:

2006

主题

2121

帖子

21万

积分

超级版主

Rank: 8Rank: 8

积分
211127
发表于 2024-4-20 16:44:04 | 显示全部楼层 |阅读模式
  1. Javascript 中 var 和 let 、const 的区别及使用方法
  2. 更新时间:2023年01月13日 10:30:48   作者:zz_sa  
  3. ECMAScript 变量是松散类型的,意思是变量可以用于保存任何类型的数据,每个变量只不过是一个用于保存任意值的命名占位符,这篇文章主要介绍了Javascript 中 var 和 let 、const 的区别以及具体使用效果,需要的朋友可以参考下
  4. 1.var、let、const简介
  5. var 声明
  6. let 声明
  7. const 声明
  8. 总结

  9. 1.var、let、const简介
  10.         ECMAScript 变量是松散类型的,意思是变量可以用于保存任何类型的数据。每个变量只不过是一个用于保存任意值的命名占位符。有三个关键字可以声明变量:var、let、const。其中 var 在ECMAScript 的所有版本中都可以使用,而 const 和 let 只能在ECMAScript 6 及更高版本中使用。


  11. var 声明
  12.          var 关键字在定义变量中定义和使用中不严格。

  13. 1.定义单个变量和同时定义多个变量,在定义多个变量时用逗号隔开。两者不赋值操作。

  14. 1
  15. 2
  16. 3
  17. 4
  18. 5
  19. 6
  20. 7
  21. 8
  22. <script>
  23.     var userName;
  24.     var age,address,phone;
  25.     console.log(userName)
  26.     console.log(age)
  27.     console.log(address)
  28.     console.log(phone)
  29. </script>
  30.         显然这样输出的值为 undefined 类型。



  31. 2.定义单个变量和同时定义多个变量,在定义多个变量时用逗号隔开。两者赋值操作。

  32. 1
  33. 2
  34. 3
  35. 4
  36. 5
  37. 6
  38. 7
  39. 8
  40. <script>
  41.     var userName = 'sa'
  42.     var age = 18,address = '南昌',phone = '19238824523'
  43.     console.log(userName)
  44.     console.log(age)
  45.     console.log(address)
  46.     console.log(phone)
  47. </script>


  48. 3.var重复声明变量和赋值

  49.         在 ECMAScript 中 var 关键字声明的变量可以重复声明,但是赋值操作会覆盖前面已经给变量赋的值。值不仅可以改变数据也可以改变相关的数据类型。

  50. 1
  51. 2
  52. 3
  53. 4
  54. 5
  55. 6
  56. <script>
  57. var userName = 'sa'
  58. console.log(userName)
  59. var userName = 77  // 这边也可以省略var
  60. console.log(userName)
  61. </script>


  62. 4.var 声明的作用域问题

  63.         var声明的全局变量全局作用域会挂载到 window 对象上,可以使用window. 的形式访问该变量,或者直接使用变量名的方式。在函数体内部的用var声明的变量是局部变量,当省略 va r时,当前变量会定义为全局变量,但是当我要得到这个变量的值时,我们需要先执行一下函数。

  64. 1
  65. 2
  66. 3
  67. 4
  68. 5
  69. 6
  70. 7
  71. 8
  72. 9
  73. 10
  74. 11
  75. 12
  76. <script>
  77.     // 定义函数,测试两个变量是否在函数体外部可以使用
  78.     function test(){
  79.         var msg = '这是局部变量'
  80.         message = '这是全局变量'
  81.   
  82.     }
  83.     test()
  84.     console.log(window.message)
  85.     console.log(message)
  86.     console.log(msg)
  87. </script>
  88.         输出结果如下:message 可以获得,但是 msg 不行。关键问题在于,使用 var 声明的变量会成为包含它在函数的局部变量。在一个函数内部定义了一个变量,就意味着该变量将在函数退出时被销毁。



  89. 5.var 声明提升

  90.         “提升”,也就是把所有的变量声明都拉到函数作用域的顶部。此外,反复多次使用 var 声明同一个变量也没有问题。

  91. 1
  92. 2
  93. 3
  94. 4
  95. 5
  96. 6
  97. 7
  98. 8
  99. 9
  100. <script>
  101.     // 定义函数,测试变量提升
  102.     function test(){
  103.         console.log(msg)
  104.         var msg = '变量'
  105.         console.log(msg)
  106.     }
  107.     test()
  108. </script>
  109.         正常的我们会认为,变量未定义先使用,会报一个 ~~ is not defined 的错误。但是 var 存在变量提升的行为,以上代码等同于:

  110. 1
  111. 2
  112. 3
  113. 4
  114. 5
  115. 6
  116. 7
  117. function test(){
  118.        var msg
  119.        console.log(msg)
  120.         msg = '变量'
  121.        console.log(msg)
  122.    }
  123.    test()
  124. 两者的输出结果都是如下:




  125. let 声明
  126.         let和var的作用差不多,但是有着非常重要的区别。最明显的区别就是,let 声明的范围是块作用域,而 var 声明的范围是函数作用域。

  127.         代码块由一个左花括号( { )和一个右花括号( } )标识结束。

  128. 1.简单示意 let 和 var 的区别

  129. 1
  130. 2
  131. 3
  132. 4
  133. 5
  134. 6
  135. 7
  136.     // let 和 var 的作用域区别
  137. if (true) {
  138.     var username = 'sa'
  139.     let msg = 'ss'
  140. }
  141.     console.log(username)
  142.     console.log(msg)
  143.         在当前代码中,let 定义的 msg 只作用与 if 的代码块中,在外部无法使用 msg 该变量。



  144. 2. let 不可以重复声明变量,var 可以重复声明变量

  145. 1
  146. 2
  147. 3
  148. 4
  149. 5
  150. 6
  151. <script>
  152. var userName
  153. var userName
  154. let msg
  155. let msg
  156. </script>
  157.         这里提示 msg 重复定义:



  158. 3.将 let 和 var 混合重复定义一个变量

  159. 1
  160. 2
  161. 3
  162. 4
  163. 5
  164. 6
  165. 7
  166. 8
  167. 9
  168. <script>
  169. var userName
  170. let userName
  171.   
  172. let msg
  173. var msg
  174. // 这边将会报一个重复定义的错误
  175.   
  176. </script>


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

  178. 1
  179. 2
  180. 3
  181. 4
  182. 5
  183. 6
  184. 7
  185. 8
  186. <script>
  187. let age = 20
  188. console.log(age) // 20
  189. if (true) {
  190.     let age = 18
  191.     console.log(age) // 18
  192. }
  193. </script>
  194. 4.let 声明的变量不会在作用域中被提升

  195. 1
  196. 2
  197. 3
  198. 4
  199. 5
  200. 6
  201. 7
  202. <script>
  203. function test() {
  204.     console.log(age)
  205.     let age = 20
  206. }
  207. test()
  208. </script>


  209.          let 声明的变量不会在作用域中被提升是和 var 一个很重要的区别。所以 let 声明变量,必须要先声明后使用,否则报错。

  210.         在解析代码时,JavaScript 引擎也会注意在块后面的 let 声明,只不过在此之前不能以任何方式来引用未声明的变量。在 let 声明之前的执行瞬间被称为“暂时性死区”,在此阶段引用任何后面才声明的变量都会抛出 ReferenceError。

  211. 5.let的全局声明不会成为 window 对象的属性( var 声明的变量则会)

  212. 1
  213. 2
  214. 3
  215. 4
  216. 5
  217. 6
  218. <script>
  219. var msg = '我是var声明的变量'
  220. let mess = '我是let声明的变量'
  221. console.log(window.msg)
  222. console.log(window.mess)
  223. </script>


  224. 6.条件声明

  225.         在使用 var 声明变量时,由于声明会被提升,JavaScript 引擎会自动将多余的声明在作用域顶部合并为一个声明。因为 let 的作用域是块,所有不可能检查是否已经使用了 let 声明过同名变量,同时也就不可能在没有声明的情况下使用它。

  226. 错误示范如下:

  227. 1
  228. 2
  229. 3
  230. 4
  231. 5
  232. 6
  233. 7
  234. 8
  235. 9
  236. <script>
  237.     // 当前声明两个变量
  238.     var msg = '我是var声明的变量'
  239.     let mess = '我是let声明的变量'
  240. </script>
  241. <script>
  242.     var msg = 'sa' // 这里没问题,var 存在变量提升声明来处理
  243.     let mess = 'ss' // 当之前声明了变量mess,这里将报错,let 不存在变量提升
  244. </script>


  245. 1
  246. 2
  247. 3
  248. 4
  249. 5
  250. 6
  251. 7
  252. 8
  253. 9
  254. 10
  255. 11
  256. 12
  257. 13
  258. 14
  259. 15
  260. 16
  261. 17
  262. 18
  263. <script>
  264. let mess // 当前只声明了一个 let 变量
  265. </script>
  266. <script>
  267.   if (typeof mess === 'undefined'){
  268.       let mess
  269.   }
  270.   mess = 'sa' // mess 被限制在 if {} 作用域块中,当前赋值为全局赋值
  271.   console.log(mess) // sa
  272.   try {
  273.       console.log(age)
  274.   }
  275.   catch (error) {
  276.       let age
  277.   }
  278.   age = 20  // age 被限制在 catch {} 作用域块中,当前赋值为全局赋值
  279.   console.log(age) // 20
  280. </script>


  281. 注:条件声明比较难以理解,这是一种反模式。这会让我们的程序变得更加难理解。

  282. 7.for循环中的 let 声明

  283.         当我们在用 var 时,for循环定义的迭代变量会渗透到循环体外部。

  284. 1
  285. 2
  286. 3
  287. 4
  288. for (var i = 0; i < 6; i++) {
  289.     setTimeout(() => {
  290.         console.log(i) },0) //  这将打印六个 6
  291. }


  292.         这种原因的出现情况是因为由于 var 声明的变量不存在块级作用域在退出循环时,迭代变量保存的是导致循环退出的值:6。在之后执行超时逻辑时,所有的 i 都是同一个变量,最终导致输出的值相同。

  293.         使用 let 声明 i 时:

  294. 1
  295. 2
  296. 3
  297. 4
  298. for (let i = 0; i < 6; i++) {
  299.     setTimeout(() => {
  300.         console.log(i) },0) //  这将打印六个 6
  301. }


  302.         当使用let 声明变量时,JavaScript 引擎在后台会为每一个迭代循环声明一个新的迭代变量,每个 setTimeout 引用的都是不同的变量实例,所以输出的值不同也就是我们期望中的值,也是循环执行过程中每一个迭代变量的值。

  303.         在我们对dom节点使用for循环绑定事件时,一般也是用 let 声明迭代变量。


  304. const 声明
  305.          const 的行为和 let 基本相同,唯一一个重要的区别就是它在声明变量的同时必须初始化变量,且尝试修改 const 声明的变量会导致运行时错误。

  306. 1.const 声明的变量不允许修改

  307. 1
  308. 2
  309. 3
  310. 4
  311. <script>
  312. const age = 20
  313. age = 30
  314. </script>


  315. 2.const 也是不可重复声明

  316. 1
  317. 2
  318. 3
  319. 4
  320. <script>
  321. const age = 20
  322. const age = 30
  323. </script>


  324. 3.const 声明的作用域也是块

  325. 1
  326. 2
  327. 3
  328. 4
  329. 5
  330. 6
  331. <script>
  332. if (true) {
  333.     const age = 30
  334. }
  335. console.log(age)
  336. </script>


  337. 4.const 中注意的点

  338.          const 声明限制只适用于它指向变量的引用。换句话说,如果 const 变量引用的是一个对象,那么修改这个对象内部的值不违反 const 的限制。引用未发生改变就行。

  339. 例如:修改了对象中 name 的值但不会报错。

  340. 1
  341. 2
  342. 3
  343. 4
  344. 5
  345. 6
  346. 7
  347. const student = {
  348.     name: 'sa',
  349.     age: 18,
  350.     address: 'shanghai'
  351. }
  352. student.name = 'saa'
  353. console.log(student.name) // 输出saa
  354.         如果你想用const 声明一个不会被修改的 for 循环变量,那也是可以的。也就是说,每次迭代只是创建一个新变量。这对 for-in 和 for-of 循环都是有意义的。

  355. 例如:

  356. 1
  357. 2
  358. 3
  359. for (const key in { a: 1, b: 2, c: 3 }){
  360.     console.log(key)
  361. }



  362. 总结
  363. 在我们日常使用变量声明时,遵循一下几点:

  364.         1.不使用 var

  365.         2.const 优先,let 次之

  366. 到此这篇关于Javascript 中 var 和 let 、const 的区别以及具体使用效果的文章就介绍到这了,更多相关js中var和 let 、const 区别内容
复制代码


回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

QQ|Archiver|手机版|小黑屋|紫影基地

GMT+8, 2025-1-26 11:49 , Processed in 0.100419 second(s), 18 queries .

Powered by Discuz! X3.4

Copyright © 2001-2020, Tencent Cloud.

快速回复 返回顶部 返回列表