设置本地为延迟扩展。其实也就是:延迟变量,全称延迟环境变量扩展。
事件一:
@echo off set a=4 set a=5&echo %a% pause
解说:为什么是4而不是5呢?在echo之前明明已经把变量a的值改成5了。
批处理运行命令的机制:批处理读取命令时是按行读取的(另外例如for命令等,其后用一对圆括号闭合的所有语句也当作一行),在处理之前要完成必要的预处理工作,这其中就包括对该行命令中的变量赋值。我们现在分析一下例1,批处理在运行到这句“set a=5&echo %a%”之前,先把这一句整句读取并做了预处理——对变量a赋了值,那么%a%当然就是4了!(没有为什么,批处理就是这样做的。)而为了能够感知环境变量的动态变化,批处理设计了变量延迟。简单来说,在读取了一条完整的语句之后,不立即对该行的变量赋值,而会在某个单条语句执行之前再进行赋值,也就是说“延迟”了对变量的赋值。如何开启变量延迟呢?变量延迟又需要注意什么呢?
事件二:开启 变量延迟机制!
@echo off setlocal enabledelayedexpansion set a=4 set a=5&echo !a! pause
结果:5
解说:由于启动了变量延迟,得到了正确答案。变量延迟的启动语句是“setlocal enabledelayedexpansion
”,并且变量要用一对叹号“!!
”括起来(注意要用英文的叹号),否则就没有变量延迟的效果。分析一下例2,首先“setlocal
enabledelayedexpansion”开启变量延迟,然后“set a=4”先给变量a赋值为4,“set a=5&echo
!a!”这句是给变量a赋值为5并输出(由于启动了变量延迟,所以批处理能够感知到动态变化,即不是先给该行变量赋值,而是在运行过程中给变量赋值,因此此时a的值就是5了)。
事件三: (一般通过全局变量保存返回值的比较常见),当前这个示例通过变量传递来保存函数的返回值:
1 @echo off 2 ::通过给调用者传递一个变量(var),来保存函数的返回值 3 call :myFunc var 4 echo.var:%var% 5 goto :eof 6 7 :myFunc 8 set "%1=haha" 9 goto :eof
结果:var:haha
解说:通过给 myFunc 函数传递 var 一个变量,然后 函数中 通过 set "%1=haha" 给第一个参数设置值,也就是给变量设置;
事件四: 通过 设置局部的可延迟的扩展变量 来实现循环,如下循环5次;
1 @echo off 2 ::设置局部的可延迟的扩展变量a 3 setlocal enabledelayedexpansion 4 for /l %%i in ( 1,1,5 ) do ( 5 set a=%%i echo !a! 6 ) 7 pause
解释: for /l %%i in(start,step,end) do ;这个循环的语法;每次将循环变量赋值给a,并打印出来;必须要加 setlocal enabledelayedexpansion 这一句,否则 a变量 预处理时没有这个东西,会不打印或者打印 echo是关闭状态;
事件五:处理局部变量和全局变量不冲突;SETLOCAL命令能让处理器当做是局部变量,用ENDLOCAL解除局部变量。
1 @echo off 2 :: 怎么保证局部变量和全局变量不冲突,SETLOCAL命令能让处理器当做是局部变量, 3 :: 用ENDLOCAL解除局部变量。 4 :: ENDLOCAL 会被自动调用,当批处理执行到文件末尾的时候,即GOTO:EOF。 5 :: SETLOCAL可以很好的保护函数内与外面的变量不会冲突。 6 7 set "var1=i'm goloable var1!" 8 set "var2=i'm goloable var2!" 9 10 echo.var1 before:"%var1%" 11 echo.var2 before:"%var2%" 12 call :myFunc var2 13 echo.var1 after:"%var1%" 14 echo.var2 after:"%var2%" 15 pause && goto :eof 16 ::传一个参数应用进去 17 :myFunc 18 SETLOCAL 19 set "var1=呵呵呵!" 20 set "%1=%var1%" 21 echo.var1 :"local Val1 %var1%" 22 ENDLOCAL 23 goto :eof
执行结果:
解释: 可以看到 ,代码中 将 var2 引用通过参数传进去,同时在函数中将var1的变量设置为呵呵呵!,但是函数执行完后, var1,和var2都没有变动(尽管var2是以变量的形式传进去的),他们只有在属于的局部区域,赋值才有效,endlocal 执行后,他们又恢复 全局变量的值了;
事件六:怎么跳过ENDLOCAL的屏障,返回局部变量值?采用”变量扩充“,在SETLOCAL与ENDLOCAL之间的全局变量的值会备份,当退出ENDLOCAL,该值将恢复。让命令处理器来执行ENDLOCAL 和SET命令。
1 @echo off 2 :: 返回局部变量 3 :: 怎么跳过ENDLOCAL的屏障,返回局部变量值? 4 :: 采用”变量扩充“,在SETLOCAL与ENDLOCAL之间的全局变量的值会备份,当退出ENDLOCAL,该值将恢复。 5 :: 让命令处理器来执行ENDLOCAL 和SET命令。 6 7 set "aStr=Expect no changed,Even if used in function~" 8 set "var1=Expect changed" 9 echo.aStr before:%aStr% 10 echo.var1 before:"%var1%" 11 call :myFunc var1 12 echo.aStr after:%aStr% 13 echo.var1 after:"%var1%" 14 pause && goto :eof 15 16 :myFunc 17 setLocal 18 set "aStr=Try To Change!" 19 (ENDLOCAL 20 set "%1=%aStr%" 21 ) 22 goto :eof
执行结果:
解释:aStr 变量通过全局变量的形式,在 setlocal 和 endlocal 之间赋值,这个赋值肯定是不会影响外部的aStr的值的,怎样将 这个局部设置的值保存下来?如上代码: 在 setlocal 和 endlocal 之外 ,通过 将这个值设置给 传进来的参数变量、或者另一个全局变量,即可将 该局部变量的赋值保存下来并返回;如上图,var1的值变为了 局部赋值:Try To Change!;
事件七:编写递归函数,Fibonacci函数;让函数局部变量的变换对调用者是可见的,循环调用函数,让变量可重用:
1 @echo off 2 :: 编写斐波拉里 函数,用来计算多少以内的斐波拉里数! 递归! 3 set "fst=0" 4 set "fib=1" 5 set "limit=10000" 6 7 :: fib 传的是引用, 另两个传的是值; fib最后用来返回值 8 call :feberlalie fib,%fst%,%limit% 9 echo. The next Fibonacci number greater or equal %limit% is %fib%. 10 pause && goto :eof 11 12 :feberlalie 13 SETLOCAL 14 15 ::可以利用set /a 进行连续赋值,只要用逗号分开每个变量名就可以了。是下面三行代码的缩写 16 set /a "Num1=%1,Num2=%2,limit=%3" 17 :: set /a "Num1=%1" 18 :: set /a "Num2=%2" 19 :: set /a "limit=%3" 20 21 :: 在set /a 计算时,可以省略变量的%号或!号,极为方便。 是带百分号的 加法的缩写 22 :: set /a "Sum=Num1 + Num2" 23 set /a "Sum=%Num1% + %Num2%" 24 25 echo.Num1 :%Num1% 26 if /i %Sum% LSS %limit% call:feberlalie Sum,%Num1%,%limit% 27 (ENDLOCAL 28 IF "%1" NEQ "" SET "%1=%Sum%" 29 ) 30 goto :eof
执行结果:
总结:定义一个标准的dos batch script function(ps:人家总结的不错,直接拿来用了):
1 :myFunctionName -- function description here 函数名 2 :: 参数描述 -- %~1: argument description here 下面都是函数体 3 SETLOCAL 4 REM.--function body here 5 set LocalVar1=... 6 set LocalVar2=... 7 (ENDLOCAL & REM -- RETURN VALUES 待返回的值 8 IF "%~1" NEQ "" SET %~1=%LocalVar1% 9 IF "%~2" NEQ "" SET %~2=%LocalVar2% 10 ) 11 GOTO:EOF
学习参考链接: