首页 > 其他分享 >说JS作用域,就不得不说说自执行函数

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

时间:2024-03-17 15:45:10浏览次数:28  
标签:console log 作用域 JS 访问 var 函数

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

前言

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

适可而止,浅尝辄止。

JS作用域

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

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

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

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

// 全局作用域
var a = "global_var_a";
console.log("全局作用域中访问:" + a); // 全局作用域中访问:global_var_a

if (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_a

if (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_b

if (true) {
	let a = "block_let_a";
	const b = "block_const_b";
	console.log("在判断语句中访问:" + a); // 在判断语句中访问:block_let_a
	console.log("在判断语句中访问:" + b); // 在判断语句中访问:block_const_b
	
	let 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_a
	console.log("在函数中访问:" + b); // 在函数中访问:func_const_b

	let 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模块就是以这种方式提供的。

总结

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

标签:console,log,作用域,JS,访问,var,函数
From: https://www.cnblogs.com/vipygd/p/18078672

相关文章

  • 激活函数初识
    一:激活函数的概念1.激活函数的定义激活函数(ActivationFunction)是一种添加到人工神经网络中的函数,旨在帮助网络学习数据中的复杂模式。类似于人类大脑中基于神经元的模型,激活函数最终决定了要发射给下一个神经元的内容。(是一种映射,把所得到的结果转化成其他的表达的方法)2......
  • 重写len函数对象魔术方法
    ###len魔术方法'''触发时机:使用len(对象)的时候自动触发功能:用于检测对象中或者类中某个内容的个数参数:一个self接受当前对象返回值:必须返回整型'''len(对象)=>类中的所有自定义成员classMyClass():pty1=1pty2=2__pty3=3deffunc1(): passdeffunc2(): ......
  • python一些函数用法
    1、性别defxb(x):ifpd.isnull(x):return"空"elifint(x[-2])%2==1:return"男"elifint(x[-2])%2==0:return"女"else:return"其他"deff(x):returnx.nunique()2、省份defprovince(x):ifstr(x)[0:2]=='11':......
  • call仿函数
    call仿函数####__call__魔术方法''' 触发时机:把对象当作函数调用的时候自动触发 功能:模拟函数化操作 参数:参数不固定,至少一个self参数 返回值:看需求'''#(1)基本语法classMyClass(): def__call__(self): print("__call__魔术方法被触发...")obj=M......
  • js面试(防抖)
    一、什么是防抖防抖(Debounce)是一种用于减少特定事件触发频率的技术。在编程中,它通常用于确保函数或方法不会在很短的时间内被频繁调用,这有助于优化性能并避免不必要的计算或操作。防抖的实现原理是,在事件被触发后,一个定时器会被设置。如果在定时器完成之前,相同的事件再次被触发,......
  • 全栈的自我修养 ———— js如何处理并发的大量请求??
    假如一个事件段内传过来一百多个请求,我们该处理大量并发请求呢过程实现验证假如现在允许的并发量为4,有一百个请求!假如现在允许的并发量为1,有一百个请求!源码过程实现定义一个方法,这个方法会放入你在这个时间段接收到的请求,这里100位例!consturlArr=[]......
  • del析构函数
    del析构函数####__del__魔术方法(析构方法)''' 触发时机:当对象被内存回收的时候自动触发[1.页面执行完毕回收所有变量2.所有对象被del的时候]功能:对象使用完毕后资源回收 参数:一个self接受对象 返回值:无'''#(1)基本语法classLion(): def__init__(self,na......
  • 微信小程序uniapp+vue+nodejs宝宝成长记录系统的设计与实现
    本文先通过对相关系统的调研,提出开发基于微信小程序的宝宝成长记录系统的意义,然后使用当前主流的技术进行开发,满足基于微信小程序的宝宝成长记录系统的技术要求,分析系统需要实现的功能并进行设计。梳理业务流程,并根据功能设计数据库,最后通过编码实现,介绍实现的关键算法逻辑。在......
  • 【滤波跟踪】基于Huber函数和最大相关熵的抗差滤波算法实现GNSS导航定位粗差处理结合
     ✅作者简介:热爱科研的Matlab仿真开发者,修心和技术同步精进,代码获取、论文复现及科研仿真合作可私信。......
  • UG NX二次开发(C#)-CAM-采用内部函数获取drill操作的进给
    CAM二次开发第一章UGNX二次开发(C#)-CAM-采用内部函数获取drill操作的进给文章目录CAM二次开发1、前言2、在高版本中打开drill加工模板3、读取drill加工操作3.1用NXOpen的FindObject实现3.2、通过Collection集合获取3.3通过选择工序导航器实现(Ufun和NXOpen......