说JS作用域,就不得不说说自执行函数

2024-03-18 01:44

本文主要是介绍说JS作用域,就不得不说说自执行函数,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

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

前言

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

适可而止,浅尝辄止。

JS作用域

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

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

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

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

// 全局作用域
var a = "global_var_a";
console.log("全局作用域中访问:" + a); // 全局作用域中访问:global_var_aif (true) {console.log("在判断语句中访问:" + a); // 在判断语句中访问:global_var_a
}function getA() {console.log("在函数中访问:" + a); // 在函数中访问:global_var_a
}getA()// ==================================================================================
// 函数作用域
var a = "global_var_a";
console.log("全局作用域访问:" + a); // 全局作用域访问:global_var_aif (true) {var a = "block_var_a"; // 与全局变量同名console.log("在判断语句中访问:" + a); // 在判断语句中访问:block_var_a
}function getA() {var a = "func_var_a"; // 与全局变量同名var b = "func_var_b";console.log("在函数中访问:" + a); // 在函数中访问:func_var_a
}getA()
console.log("在全局作用域中访问:" + a); // 在全局作用域中访问:block_var_a;由于允许变量重复声明,导致变量被覆盖
console.log("在全局作用域中访问:" + b); // Uncaught ReferenceError: b is not defined// ==================================================================================
// 块作用域
var a = "global_var_a";
const b = "global_const_b";console.log("全局作用域中访问:" + a); // 全局作用域中访问:global_var_a
console.log("全局作用域中访问:" + b); // 全局作用域中访问:global_const_bif (true) {let a = "block_let_a";const b = "block_const_b";console.log("在判断语句中访问:" + a); // 在判断语句中访问:block_let_aconsole.log("在判断语句中访问:" + b); // 在判断语句中访问:block_const_blet c = "block_let_c";const d = "block_let_d";
}function getA() {let a = "func_let_a";const b = "func_const_b";console.log("在函数中访问:" + a); // 在函数中访问:func_let_aconsole.log("在函数中访问:" + b); // 在函数中访问:func_const_blet e = "func_let_e";const f = "func_const_f";
}getA()
console.log("全局作用域中访问:" + a); // 全局作用域中访问:global_var_a
console.log("全局作用域中访问:" + b); // 全局作用域中访问:global_const_b
// console.log("全局作用域中访问:" + c); Uncaught ReferenceError: c is not defined
// console.log("全局作用域中访问:" + d); Uncaught ReferenceError: d is not defined
// console.log("全局作用域中访问:" + e); Uncaught ReferenceError: e is not defined
// console.log("全局作用域中访问:" + f); Uncaught ReferenceError: f is not defined

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

if (false) {var a = "abc";console.log(a);
} else {console.log(a);
}
console.log(a);

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

JS自执行函数

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

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

自执行函数有三种写法:

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

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

var myModule = (function(){var privateVar ='私有变量';function privateFunc(){console.log('私有函数');}return {publicFunc: function() {console.log('公有函数');}};
})();myModule.publicFunc(); // "公有函数"
console.log(myModule.privateVar); // undefined
myModule.privateFunc(); // Uncaught TypeError: myModule.privateFunc is not a function

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

总结

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

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

在这里插入图片描述

这篇关于说JS作用域,就不得不说说自执行函数的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/820813

相关文章

Kotlin 作用域函数apply、let、run、with、also使用指南

《Kotlin作用域函数apply、let、run、with、also使用指南》在Kotlin开发中,作用域函数(ScopeFunctions)是一组能让代码更简洁、更函数式的高阶函数,本文将... 目录一、引言:为什么需要作用域函数?二、作用域函China编程数详解1. apply:对象配置的 “流式构建器”最

用js控制视频播放进度基本示例代码

《用js控制视频播放进度基本示例代码》写前端的时候,很多的时候是需要支持要网页视频播放的功能,下面这篇文章主要给大家介绍了关于用js控制视频播放进度的相关资料,文中通过代码介绍的非常详细,需要的朋友可... 目录前言html部分:JavaScript部分:注意:总结前言在javascript中控制视频播放

Spring定时任务只执行一次的原因分析与解决方案

《Spring定时任务只执行一次的原因分析与解决方案》在使用Spring的@Scheduled定时任务时,你是否遇到过任务只执行一次,后续不再触发的情况?这种情况可能由多种原因导致,如未启用调度、线程... 目录1. 问题背景2. Spring定时任务的基本用法3. 为什么定时任务只执行一次?3.1 未启用

Android Kotlin 高阶函数详解及其在协程中的应用小结

《AndroidKotlin高阶函数详解及其在协程中的应用小结》高阶函数是Kotlin中的一个重要特性,它能够将函数作为一等公民(First-ClassCitizen),使得代码更加简洁、灵活和可... 目录1. 引言2. 什么是高阶函数?3. 高阶函数的基础用法3.1 传递函数作为参数3.2 Lambda

C++中::SHCreateDirectoryEx函数使用方法

《C++中::SHCreateDirectoryEx函数使用方法》::SHCreateDirectoryEx用于创建多级目录,类似于mkdir-p命令,本文主要介绍了C++中::SHCreateDir... 目录1. 函数原型与依赖项2. 基本使用示例示例 1:创建单层目录示例 2:创建多级目录3. 关键注

C++中函数模板与类模板的简单使用及区别介绍

《C++中函数模板与类模板的简单使用及区别介绍》这篇文章介绍了C++中的模板机制,包括函数模板和类模板的概念、语法和实际应用,函数模板通过类型参数实现泛型操作,而类模板允许创建可处理多种数据类型的类,... 目录一、函数模板定义语法真实示例二、类模板三、关键区别四、注意事项 ‌在C++中,模板是实现泛型编程

kotlin的函数forEach示例详解

《kotlin的函数forEach示例详解》在Kotlin中,forEach是一个高阶函数,用于遍历集合中的每个元素并对其执行指定的操作,它的核心特点是简洁、函数式,适用于需要遍历集合且无需返回值的场... 目录一、基本用法1️⃣ 遍历集合2️⃣ 遍历数组3️⃣ 遍历 Map二、与 for 循环的区别三、高

C语言字符函数和字符串函数示例详解

《C语言字符函数和字符串函数示例详解》本文详细介绍了C语言中字符分类函数、字符转换函数及字符串操作函数的使用方法,并通过示例代码展示了如何实现这些功能,通过这些内容,读者可以深入理解并掌握C语言中的字... 目录一、字符分类函数二、字符转换函数三、strlen的使用和模拟实现3.1strlen函数3.2st

MySQL中COALESCE函数示例详解

《MySQL中COALESCE函数示例详解》COALESCE是一个功能强大且常用的SQL函数,主要用来处理NULL值和实现灵活的值选择策略,能够使查询逻辑更清晰、简洁,:本文主要介绍MySQL中C... 目录语法示例1. 替换 NULL 值2. 用于字段默认值3. 多列优先级4. 结合聚合函数注意事项总结C

Java8需要知道的4个函数式接口简单教程

《Java8需要知道的4个函数式接口简单教程》:本文主要介绍Java8中引入的函数式接口,包括Consumer、Supplier、Predicate和Function,以及它们的用法和特点,文中... 目录什么是函数是接口?Consumer接口定义核心特点注意事项常见用法1.基本用法2.结合andThen链