经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » JS/JS库/框架 » JavaScript » 查看文章
说JS作用域,就不得不说说自执行函数
来源:cnblogs  作者:^_^果冻^_^  时间:2024/3/18 15:05:50  对本文有异议

一个兜兜转转,从“北深”回到三线城市的小码农,热爱生活,热爱技术,在这里和大家分享一个技术人员的点点滴滴。欢迎大家关注我的微信公众号:果冻想

前言

不得不吐槽,学个JS,这个概念也太多了,但是这些概念你不懂吧,代码你都看不懂,你都寸步难行。好吧,这又遇到了作用域方面的知识盲区,然后发现,又牵扯出了自执行函数。那又能咋整,为了这点破工资,学呗。

适可而止,浅尝辄止。

JS作用域

作用域指的是一个变量的作用范围。我们定义的变量它只能在自己的作用域内有效,超出了自己的作用域,变量就不起作用了。但是,JavaScript这门语言很活,如果你不搞懂它的作用域原理,你很可能在不知不觉中被坑了。

在JavaScript中,主要有三种作用域:

  1. 全局作用域:在所有函数外部定义的变量、函数和对象,可以被代码中的所有部分访问。
  2. 函数作用域:在函数内部定义的变量、函数和对象,只能在函数内部访问。
  3. 块级作用域:在块级作用域(使用 let 或 const 关键字定义的变量)中定义的变量,只能在该块内访问。

下面通过不同的示例代码来演示这几种作用域,以便更好的理解:

  1. // 全局作用域
  2. var a = "global_var_a";
  3. console.log("全局作用域中访问:" + a); // 全局作用域中访问:global_var_a
  4. if (true) {
  5. console.log("在判断语句中访问:" + a); // 在判断语句中访问:global_var_a
  6. }
  7. function getA() {
  8. console.log("在函数中访问:" + a); // 在函数中访问:global_var_a
  9. }
  10. getA()
  11. // ==================================================================================
  12. // 函数作用域
  13. var a = "global_var_a";
  14. console.log("全局作用域访问:" + a); // 全局作用域访问:global_var_a
  15. if (true) {
  16. var a = "block_var_a"; // 与全局变量同名
  17. console.log("在判断语句中访问:" + a); // 在判断语句中访问:block_var_a
  18. }
  19. function getA() {
  20. var a = "func_var_a"; // 与全局变量同名
  21. var b = "func_var_b";
  22. console.log("在函数中访问:" + a); // 在函数中访问:func_var_a
  23. }
  24. getA()
  25. console.log("在全局作用域中访问:" + a); // 在全局作用域中访问:block_var_a;由于允许变量重复声明,导致变量被覆盖
  26. console.log("在全局作用域中访问:" + b); // Uncaught ReferenceError: b is not defined
  27. // ==================================================================================
  28. // 块作用域
  29. var a = "global_var_a";
  30. const b = "global_const_b";
  31. console.log("全局作用域中访问:" + a); // 全局作用域中访问:global_var_a
  32. console.log("全局作用域中访问:" + b); // 全局作用域中访问:global_const_b
  33. if (true) {
  34. let a = "block_let_a";
  35. const b = "block_const_b";
  36. console.log("在判断语句中访问:" + a); // 在判断语句中访问:block_let_a
  37. console.log("在判断语句中访问:" + b); // 在判断语句中访问:block_const_b
  38. let c = "block_let_c";
  39.     const d = "block_let_d";
  40. }
  41. function getA() {
  42. let a = "func_let_a";
  43. const b = "func_const_b";
  44. console.log("在函数中访问:" + a); // 在函数中访问:func_let_a
  45. console.log("在函数中访问:" + b); // 在函数中访问:func_const_b
  46. let e = "func_let_e";
  47.     const f = "func_const_f";
  48. }
  49. getA()
  50. console.log("全局作用域中访问:" + a); // 全局作用域中访问:global_var_a
  51. console.log("全局作用域中访问:" + b); // 全局作用域中访问:global_const_b
  52. // console.log("全局作用域中访问:" + c); Uncaught ReferenceError: c is not defined
  53. // console.log("全局作用域中访问:" + d); Uncaught ReferenceError: d is not defined
  54. // console.log("全局作用域中访问:" + e); Uncaught ReferenceError: e is not defined
  55. // console.log("全局作用域中访问:" + f); Uncaught ReferenceError: f is not defined

这里顺便多说一嘴,关于var定义变量时的变量提升问题,看下面这段代码:

  1. if (false) {
  2. var a = "abc";
  3. console.log(a);
  4. } else {
  5. console.log(a);
  6. }
  7. console.log(a);

我们执行上面的代码,理应报Uncaught ReferenceError: a is not defined这个错误的,但是由于变量提升问题,这段代码是不会报错的,但是逻辑是有问题的。

JS自执行函数

说完JS的作用域问题,再来说说自执行函数。它的定义如下:

自执行函数是指定义后立即执行的函数,它可以被用来创建一个私有作用域。自执行函数的作用域只在函数内部有效,可以用来隐藏变量和函数,避免全局命名冲突,保持代码的整洁性和可维护性。它可以用来创建私有作用域、实现模块化、简化代码等等,非常灵活和实用。

自执行函数有三种写法:

  1. (function("参数") {"函数方法";})("给参数传的值")
  2. (function("参数") {"函数方法";}("给参数传的值"))
  3. !function("参数") {"函数方法";}("给参数传的值") // ! 可以换作 void 或其他运算符(比如 +,-,= 等,都能起到立即执行的作用)

因为全局变量很容易引起一些Bug,所以使用自执行函数来实现模块化,内部变量和函数对外部不可见,只有暴露出去的接口可以被外部访问。看下面这段代码。

  1. var myModule = (function(){
  2. var privateVar ='私有变量';
  3. function privateFunc(){
  4. console.log('私有函数');
  5. }
  6. return {
  7. publicFunc: function() {
  8. console.log('公有函数');
  9. }
  10. };
  11. })();
  12. myModule.publicFunc(); // "公有函数"
  13. console.log(myModule.privateVar); // undefined
  14. myModule.privateFunc(); // Uncaught TypeError: myModule.privateFunc is not a function

在上面的代码中,自执行函数返回一个包含公有函数publicFunc的对象,这个函数可以被外部访问,而私有变量privateVar和私有函数privateFunc对外部不可见。这样可以有效地隔离代码,避免全局变量污染,提高代码的可维护性和重用性。大部分开元的JavaScript模块就是以这种方式提供的。

总结

每天一个小知识点,每天进步一点,与君共勉。

原文链接:https://www.cnblogs.com/vipygd/p/18078672

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号