【转】BAT 延迟变量

延迟环境变量在bat里是重中之重,虽然前面说过,熟练应用for才算会写批处理,但如果不懂延迟环境变量的话,那么你就只能写出简单的批处理,而for语句也不能发挥最大的作用。

延迟环境变量在cmd下默认是关闭的,如果要使用延迟环境变量,可以用以下两种方法开启:

1.      cmd下开启/关闭延迟环境变量

cmd下开启或延迟环境变量用/v参数:

cmd /v:on    //开启延迟环境变量

cmd /v:off   //关闭延迟环境变量

2.      批处理中开启/关闭延迟环境变量

在批处理中开启或关闭延迟环境变量用setlocal命令:

setlocal EnableDelayedExpansion    //开启延迟环境变量

setlocal DisableDelayedExpansion   //关闭延迟环境变量

延迟环境变量到底有什么用呢?这里我们就要从cmd命令解释器工作原理来说起了。

当我们准备执行一条命令的时候,命令解释器会先将命令读取,如果命令中有环境变量,那么就会将变量的值先读取来出,然后在运行这条命令,如:echo %windir%,当我们执行这条命令的时候,命令解释器会先读出%windir%的值,即c:\windows,然后执行echo,得到的结果是屏幕上显示出c:\windows。这个应该很好理解。

但是,有的时候,我们在执行一条命令的时候,命令解释器将环境变量的值读出来以后,我们的环境变量的值发生了改变,这时个再执行命令就是使用的变量改变前的值,这就不是我们想要的结果了,举个例子:

cmd下运行依次运行下面的两条命令:

set a=1
set /a a+=1 > nul & echo %a%

运行后我们会发现,echo 得到的结果是1,而不是我们想像中的2,这时候我们可以直接输入set回车,就可以看到a的值是2,但为什么显示出来的是1而不是2呢?这就是因为在准备执行上面第二句set /a a+=1 > nul & echo %a%的时候,命令解释器先将a的值读取出来了,这时还没有执行set /a a+=1,所以a的值是1,然后将1给了set,同时也给了echo,所以,虽然计算后,a=2,但显示出来的还是a=1。这时,如果我们想得到我们想要的结果,那么就要开启延迟环境变量了。

延迟环境变量,顾名思义,延迟读取了环境变量。当我们开启延迟环境变量之后,命令解释器就不再是运行命令前先将环境变量读取出来再执行,而是当我们用到延迟环境变量的时候再去从环境变量中读取出来,这就延迟了环境变量读取的时间。

还是上面的例子,我们开启延迟环境变量后再次运行:

cmd /v:on
set a=1
set /a a+=1 > nul & echo !a!

在第一句开启延迟环境变量后,运行第三句的时候就不是先将a的值读出来,而是在使用到的时候再读:运行到set /a的时候读出a=1,执行后a=2,在执行echo的时候再去读取a的值a=2,所以最后的结果就是我们想要的2了。

我们注意到,在这里调用环境变量a不是用的%a%而是用的!a!。这是因为,我们用命令开启了延迟环境变量,只是允许我们使用延迟环境变量,并不是所有变量的应用是延迟读取,这时我们也可以用%a%以正常的方式读取环境变量,如果我们想使环境变量延迟读取,我们就要用到!a!来代替%a%

延迟环境变量在批处理中应用很广泛,想要作出功能强大的批处理,有时延迟环境变量是不可缺少的。而应用最多的地方就是在forif语句里,因为forif作为一个语句,后面可以跟许多命令,不管跟多少命令,但他还是一条命令,所以说在forif中应用的最多。举个例子:

@echo off & setlocal EnableDelayedExpansion
set a=0
for /l %%i in (1,1,10) do (
set /a a=%%i%%2
if !a!==0 echo 1-10的偶数有:%%i)
pause

上面的代码是输出110中能被2整除的数(在批处理中取余运算符要用%%代替命令提示符下的%),虽然if是单独的一行,没有和set在一行,由于有“()”括起来,他只是for语句中的一部分。

如果我们将!a!换成%a%,那么由于a初始值是0,那么运行for的时候就会将a的值读出来,虽然开启了延迟环境变量,但在if的时候就不会再次读取a的值了,由于0==0永远成立,那么结果就会将1-10所有值全部例出来。

 

原文链接:http://hi.baidu.com/ynnal911/blog/item/2e91f91bd7c6e50a34fa41d6.html

posted on 2015-12-28 10:42  是知也  阅读(798)  评论(0编辑  收藏  举报

导航