与Bat脚本的故事
因为工作时需要将定时处理的业务抽出来,废弃通过监听定时调用的这种方法,改为通过第三方软件定时执行bat脚本来实现,所以学习了一下bat脚本,整理出一些学习中的基础点和重点。
基础点:
(1)bat脚本的标签
each和@:@强制关闭命令回显,无论each是否开启
each:打开回显/关闭回显(each on/off)
输出信息(each)
输出空行(each .)
建立新文件/增加文件内容 (ECHO 文件内容>文件名/ECHO 文件内容>>文件名)
REM和:加非字母、数字的特殊符号:REM对程序进行注释,注释后的程序不运行,但会有回显
:加非字母、数字的特殊符号对程序进行注释,注释后的程序被视为无效的命令,任何情况都没有回显
pause:暂停命令,暂停后按任意键继续
errorlevel:程序结束后,查看返回码
tittle:设置CMD窗口标题
color:设置窗口的颜色
mode:系统配置
goto和:加标签 :进行跳转,通过:加标签来找到要跳转的下一步位置
find:文件中搜寻字符串
star:批处理时调用外部程序的命令
call;批处理的过程中调用另一个批处理命令,处理完后继续处理之前的批处理
if:判断语句
setlocal与延迟变量:本条内容引用[英雄出品]的批处理教程:
例1:
1
2
3
4
|
@ echo off set a=4 set a=5 & echo %a% pause |
结果:4
解说:为什么是4而不是5呢?在echo之前明明已经把变量a的值改成5了?
让我们先了解一下批处理运行命令的机制:
批 处理读取命令时是按行读取的(另外例如for命令等,其后用一对圆括号闭合的所有语句也当作一行),在处理之前要完成必要的预处理工作,这其中就包括对该 行命令中的变量赋值。我们现在分析一下例1,批处理在运行到这句“set a=5 & echo %a%”之前,先把这一句整句读取并做了预处理——对变量a赋了值,那么%a%当然就是4了!(没有为什么,批处理就是这样做的。)
而为了能够感知环境变量的动态变化,批处理设计了变量延迟。简单来说,在读取了一条完整的语句之后,不立即对该行的变量赋值,而会在某个单条语句执行之前再进行赋值,也就是说“延迟”了对变量的赋值。
那么如何开启变量延迟呢?变量延迟又需要注意什么呢?举个例子说明一下:
例2:
1
2
3
4
5
|
@ 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了)。
再举一个例子巩固一下。
例3:
1
2
3
4
5
6
7
|
@ echo off setlocal enabledelayedexpansion for /l %%i in (1,1,5) do ( set a=%%i echo !a! ) pause |
结果:
1
2
3
4
5
解说:本例开启了变量延迟并用“!!”将变量扩起来,因此得到我们预期的结果。如果不用变量延迟会出现什
么结果呢?结果是这样的:
ECHO 处于关闭状态。
ECHO 处于关闭状态。
ECHO 处于关闭状态。
ECHO 处于关闭状态。
ECHO 处于关闭状态。
即没有感知到for语句中的动态变化。
提示:在没有开启变量延迟的情况下,某条命令行中的变量改变,必须到下一条命令才能体现。这一点也可以加以利用,看例子。
例:交换两个变量的值,且不用中间变量
1
2
3
4
5
6
7
8
9
10
|
@ echo off ::目的:交换两个变量的值,但是不使用临时变量 ::Code by JM 2007-1-24 [email=CMD@XP]CMD@XP[/email] set var1=abc set var2=123 echo 交换前: var1=%var1% var2=%var2% set var1=%var2%& set var2=%var1% echo 交换后: var1=%var1% var2=%var2% pause |