JavaScript 变量作用域及闭包

本文通过多个实例详细解析了JavaScript中的变量作用域规则及闭包的概念,展示了如何利用闭包保留变量状态,并讨论了其可能导致的问题。

作用域JavaScript 的变量作用域是按照函数划分的,为了快速的了解它的特性,我们通过实例来进行演示。

实例一:

<mce:script type="text/javascript"><!-- var i = 1; // 弹出内容为 1 true 的提示框 alert(window.i + ' ' + (window.i == i)); // --></mce:script>

分析:
在全局定义的变量其实就是 window 对象的属性。
上面的例子可以看到,我们定义全局变量的同时,window 对象会产生一个相应的属性,如何让我们的代码避免产生这个属性呢,看下面的例子。
实例二:

<mce:script type="text/javascript"><!-- var document = 1; window.onload = function(){ alert(document); } // 弹出内容为 1 的提示框 alert(window.document); // --></mce:script>

这种情况是我们不想看到的,我们可以这么做:

<mce:script type="text/javascript"><!-- function test(){ var document = 1; window.onload = function(){ alert(document); } } test(); // 弹出内容为 [object] 的提示框 alert(window.document); // --></mce:script>

为了使代码更加简洁,我们可以这样:

<mce:script type="text/javascript"><!-- (function(){ var document = 1; window.onload = function(){ alert(document); } })(); // 弹出内容为 [object] 的提示框 alert(window.document); // --></mce:script>

分析:
这种运行匿名方法的形式,在 JavaScript 的主流框架中经常见到,这样做可以避免产生不必要的 window 对象的属性,减少冲突的可能。
实例三:

<mce:script type="text/javascript"><!-- (function(){ if('1' == '1'){ var i = 1; } // 弹出内容为 1 的提示框 alert(i); })(); // --></mce:script>

分析:
变量的作用域是整个函数,不是{}块。
实例四:

<mce:script type="text/javascript"><!-- var i = 1; // 弹出内容为 1 的提示框 alert(i); var i = 2; // 弹出内容为 2 的提示框 alert(i); // --></mce:script>

分析:
一个变量可以被重新定义,这个看起来有些怪,因为在很多其他语言里这样是行不通的。
实例五:

<mce:script type="text/javascript"><!-- function test(){ i = 1; } test(); // 弹出内容为 1 的提示框 alert(window.i); // --></mce:script>

分析:
如果对一个没有初始化的变量进行赋值操作,那么这个变量会作为全局变量。

实例六:

<mce:script type="text/javascript"><!-- window.onload = function(){ var i = 1; function test(){ alert(i); } // 弹出内容为 1 的提示框 test(); } // --></mce:script>


分析:
内部函数可以访问外部函数的变量,这个就引出了一个新的概念,那就是闭包。
闭包
什么是闭包呢,简单的说就是一个函数 A ,它的内部函数 B 可以访问 A 内定义的变量,即使函数 A 已经终止。下面通过实例进行了解。
实例七:

<mce:script type="text/javascript"><!-- window.onload = function(){ var i = 1; window.onunload = function(){ alert(i); } } // --></mce:script>

分析:
当整个页面加载完成时,会触发 onload 事件,这个onload 事件方法里给窗口的onunload 事件注册了一个方法,这个方法里用到了onload 事件方法里声明的变量,然后onload 事件方法运行结束,这时候我们点击关闭窗口,会弹出内容为1的提示框,说明onunload 的事件方法成功的调用了onload 事件方法里声明的变量。
为了进一步了解闭包的特性,看下面的例子
实例八:

<mce:script type="text/javascript"><!-- function initX(oarg){ // 定义一个变量 var x = oarg; // 定义一个显示变量的方法 var funGet = function(){ alert(x); } // 定义一个对变量进行修改的方法 var funSet = function(iarg){ x = iarg; } // 返回这两个方法 return [funGet,funSet]; } // 运行一个方法实例,返回值为包含 get 和 set 方法的数组 var funArr = initX(1); // 得到 get 方法 var funGet = funArr[0]; // 得到 set 方法 var funSet = funArr[1]; // 运行 get 方法,显示initX方法实例内的 x 变量,结果为 1 funGet(); // 运行 set 方法,对initX方法实例内的 x 变量进行赋值 funSet(2); // 运行 get 方法,显示initX方法实例内的 x 变量,结果为 2 funGet(); // --></mce:script>

分析:
当内部函数对外部函数定义的变量进行调用时,实际上引用的是这个变量的内存块,所以当我们调用内部函数时,引用的变量值是当前这个变量的实际内容。
闭包功能虽然强大,但是如果不注意,它也会给我们带来困扰。看下面的例子。
实例九:

<button id="main">run</button> <mce:script type="text/javascript"><!-- (function(){ var obj = document.getElementById("main"); var funArr = ['onclick','onkeypress']; for(var i=0; i<funArr.length; i++){ var temp = funArr[i]; obj[temp] = function(){ alert(temp); } } })(); // --></mce:script>


写代码的原意是给 id 是 main 的按钮注册点击事件和按键事件,事件的内容是分别弹出事件名称的提示框。但是结果有点匪夷所思,两个事件的提示框全是 onkeypress,根据闭包的原理,我们仔细分析,就会发现当两个事件方法被调用时 temp 变量 指向的是 funArr[1] 的内容,我们可以这样修改来解决这个问题:

<button id="main">run</button> <mce:script type="text/javascript"><!-- (function(){ var obj = document.getElementById("main"); var funArr = ['onclick','onkeypress']; for(var i=0; i<funArr.length; i++){ (function(){ var temp = funArr[i]; obj[temp] = function(){ alert(temp); } })(); } })(); // --></mce:script>

把 for 循环内的代码放入一个函数内,这样,每循环一次都会产生一个函数实例,让函数实例记录 funArr 数组中的每个值,这样就避免了上面碰到的问题。

内容概要:本文介绍了一种基于蒙特卡洛模拟和拉格朗日优化方法的电动汽车充电站有序充电调度策略,重点针对分时电价机制下的分散式优化问题。通过Matlab代码实现,构建了考虑用户充电需求、电网负荷平衡及电价波动的数学模【电动汽车充电站有序充电调度的分散式优化】基于蒙特卡诺和拉格朗日的电动汽车优化调度(分时电价调度)(Matlab代码实现)型,采用拉格朗日乘子法处理约束条件,结合蒙特卡洛方法模拟大量电动汽车的随机充电行为,实现对充电功率和时间的优化分配,旨在降低用户充电成本、平抑电网峰谷差并提升充电站运营效率。该方法体现了智能优化算法在电力系统调度中的实际应用价值。; 适合人群:具备一定电力系统基础知识和Matlab编程能力的研究生、科研人员及从事新能源汽车、智能电网相关领域的工程技术人员。; 使用场景及目标:①研究电动汽车有序充电调度策略的设计与仿真;②学习蒙特卡洛模拟与拉格朗日优化在能源系统中的联合应用;③掌握基于分时电价的需求响应优化建模方法;④为微电网、充电站运营管理提供技术支持和决策参考。; 阅读建议:建议读者结合Matlab代码深入理解算法实现细节,重点关注目标函数构建、约束条件处理及优化求解过程,可尝试调整参数设置以观察不同场景下的调度效果,进一步拓展至多目标优化或多类型负荷协调调度的研究。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值