bat批处理中怎样⽤while循环
bat
常⽤命令
echo、@、call、pause、rem(⼩技巧:⽤::代替rem)是批处理⽂件最常⽤的⼏个命令,我们就从他们开始学起。
==== 注 ===========
⾸先, @ 不是⼀个命令, ⽽是DOS 批处理的⼀个特殊标记符, 仅⽤于屏蔽命令⾏回显. 下⾯是DOS命令⾏或批处理中可能会见到的⼀些特殊标记符:
CR(0D) 命令⾏结束符
Escape(1B) ANSI转义字符引导符
Space(20) 常⽤的参数界定符
Tab(09) ; = 不常⽤的参数界定符
+ COPY命令⽂件连接符
* ? ⽂件通配符
"" 字符串界定符
| 命令管道符
< > >>  ⽂件重定向符
@ 命令⾏回显屏蔽符
/ 参数开关引导符
: 批处理标签引导符
% 批处理变量引导符
其次, :: 确实可以起到rem 的注释作⽤, ⽽且更简洁有效; 但有两点需要注意:
第⼀, 除了 :: 之外, 任何以 :开头的字符⾏, 在批处理中都被视作标号, ⽽直接忽略其后的所有内容, 只
是为了与正常的标号相区别, 建议使⽤goto 所⽆法识别的标号, 即在 :后紧跟⼀个⾮字母数字的⼀个特殊符号.
第⼆, 与rem 不同的是, ::后的字符⾏在执⾏时不会回显, ⽆论是否⽤echo on打开命令⾏回显状态, 因为命令解释器不认为他是⼀个有效的命令⾏, 就此点来看, rem 在某些场合下将⽐ :: 更为适⽤; 另外, rem 可以⽤于 config.sys ⽂件中.
=====================
echo 表⽰显⽰此命令后的字符
echo off 表⽰在此语句后所有运⾏的命令都不显⽰命令⾏本⾝
@与echo off相象,但它是加在每个命令⾏的最前⾯,表⽰运⾏时不显⽰这⼀⾏的命令⾏(只能影响当前⾏)。
call 调⽤另⼀个批处理⽂件(如果不⽤call⽽直接调⽤别的批处理⽂件,那么执⾏完那个批处理⽂件后将⽆法返回当前⽂件并执⾏当前⽂件的后续命令)。
pause 运⾏此句会暂停批处理的执⾏并在屏幕上显⽰Press any key 的提⽰,等待⽤户按任意键后继续
rem 表⽰此命令后的字符为解释⾏(注释),不执⾏,只是给⾃⼰今后参考⽤的(相当于程序中的注释)。
==== 注 =====
此处的描述较为混乱, 不如直接引⽤个命令的命令⾏帮助更为条理
-------------------------
ECHO
当程序运⾏时,显⽰或隐藏批处理程序中的正⽂。也可⽤于允许或禁⽌命令的回显。
在运⾏批处理程序时,MS-DOS⼀般在屏幕上显⽰(回显)批处理程序中的命令。
使⽤ECHO命令可关闭此功能。
语法
ECHO [ON|OFF]
若要⽤echo命令显⽰⼀条命令,可⽤下述语法:
echo [message]
参数
ON|OFF
指定是否允许命令的回显。若要显⽰当前的ECHO的设置,可使⽤不带参数的ECHO
命令。
message
批处理文件注释指定让MS-DOS在屏幕上显⽰的正⽂。
-------------------
CALL
从⼀个批处理程序中调⽤另⼀个批处理程序,⽽不会引起第⼀个批处理的中⽌。
语法
CALL [drive:][path]filename [batch-parameters]
参数
[drive:][path]filename
指定要调⽤的批处理程序的名字及其存放处。⽂件名必须⽤.BAT作扩展名。
batch-parameters
指定批处理程序所需的命令⾏信息。
-------------------------------
PAUSE
暂停批处理程序的执⾏并显⽰⼀条消息,提⽰⽤户按任意键继续执⾏。只能在批处
理程序中使⽤该命令。
语法
PAUSE
REM
在批处理⽂件或CONFIG.SYS中加⼊注解。也可⽤REM命令来屏蔽命令(在CONFIG.SYS 中也可以⽤分号 ; 代替REM命令,但在批处理⽂件中则不能替代)。
语法
REM [string]
参数
string
指定要屏蔽的命令或要包含的注解。
指定要屏蔽的命令或要包含的注解。
=======================
例1:⽤edit编辑a.bat⽂件,输⼊下列内容后存盘为c:\a.bat,执⾏该批处理⽂件后可实现:将根⽬录中所有⽂件写⼊ a.txt中,启动UCDOS,进⼊WPS等功能。
批处理⽂件的内容为: 命令注释:
off 不显⽰后续命令⾏及当前命令⾏
dir c:\*.* & 将c盘⽂件列表写⼊a.txt
call c:\ucdos\ucdos.bat 调⽤ucdos
echo 你好 显⽰"你好"
pause 暂停,等待按键继续
rem 准备运⾏wps 注释:准备运⾏wps
cd ucdos 进⼊ucdos⽬录
wps 运⾏wps
批处理⽂件的参数
批处理⽂件还可以像C语⾔的函数⼀样使⽤参数(相当于DOS命令的命令⾏参数),这需要⽤到⼀个参数表⽰符"%"。
%[1-9]表⽰参数,参数是指在运⾏批处理⽂件时在⽂件名后加的以空格(或者Tab)分隔的字符串。变量可以从%0到%9,%0表⽰批处理命令本⾝,其它参数字符串⽤%1到%9顺序表⽰。
例2:C:根⽬录下有⼀批处理⽂件名为f.bat,内容为:
off
format %1
如果执⾏C:\>f a:
那么在执⾏f.bat时,%1就表⽰a:,这样format %1就相当于format a:,于是上⾯的命令运⾏时实际执⾏的是format a:
例3:C:根⽬录下⼀批处理⽂件名为t.bat,内容为:
off
type %1
type %2
那么运⾏C:\> b.txt
%1 : 表⽰a.txt
%2 : 表⽰b.txt
于是上⾯的命令将顺序地显⽰a.txt和b.txt⽂件的内容。
==== 注 ===============
参数在批处理中也作为变量处理, 所以同样使⽤百分号作为引导符, 其后跟0-9中的⼀个数字构成参数引⽤符. 引⽤符和参数之间 (例如上⽂中的 %1 与 a: ) 的关系类似于变量指针与变量值的关系. 当我们要引⽤第⼗⼀个或更多个参数时, 就必须移动DOS 的参数起始指针. shift 命令正充当了这个移动指针的⾓⾊, 它将参数的起始指针移动到下⼀个参数, 类似C 语⾔中的指针操作. 图⽰如下:
初始状态, cmd 为命令名, 可以⽤ %0 引⽤
cmd arg1 arg2 arg3 arg4 arg5 arg6 arg7 arg8 arg9 arg10
^ ^ ^ ^ ^ ^ ^ ^ ^ ^
| | | | | | | | | |
%0 %1 %2 %3 %4 %5 %6 %7 %8 %9
经过1次shift后, cmd 将⽆法被引⽤
cmd arg1 arg2 arg3 arg4 arg5 arg6 arg7 arg8 arg9 arg10
^ ^ ^ ^ ^ ^ ^ ^ ^ ^
| | | | | | | | | |
| | | | | | | | | |
%0 %1 %2 %3 %4 %5 %6 %7 %8 %9
经过2次shift后, arg1也被废弃, %9指向为空, 没有引⽤意义
cmd arg1 arg2 arg3 arg4 arg5 arg6 arg7 arg8 arg9 arg10
^ ^ ^ ^ ^ ^ ^ ^ ^
| | | | | | | | |
%0 %1 %2 %3 %4 %5 %6 %7 %8
遗憾的是, win9x 和DOS下均不⽀持 shift 的逆操作. 只有在 nt 内核命令⾏环境下, shift 才⽀持 /n 参数, 可以以第⼀参数为基准返复移动起始指针.
=================
特殊命令
if goto choice for是批处理⽂件中⽐较⾼级的命令,如果这⼏个你⽤得很熟练,你就是批处理⽂件的专家啦。
⼀、if 是条件语句,⽤来判断是否符合规定的条件,从⽽决定执⾏不同的命令。 有三种格式:
1、if [not] "参数" == "字符串" 待执⾏的命令
参数如果等于(not表⽰不等,下同)指定的字符串,则条件成⽴,运⾏命令,否则运⾏下⼀句。
例:if "%1"=="a" format a:
====
if 的命令⾏帮助中关于此点的描述为:
IF [NOT] string1==string2 command
在此有以下⼏点需要注意:
1. 包含字符串的双引号不是语法所必须的, ⽽只是习惯上使⽤的⼀种"防空"字符
2. string1 未必是参数, 它也可以是环境变量, 循环变量以及其他字符串常量或变量
3. command 不是语法所必须的, string2 后跟⼀个空格就可以构成⼀个有效的命令⾏
=============================
2、if [not] exist [路径\]⽂件名 待执⾏的命令
如果有指定的⽂件,则条件成⽴,运⾏命令,否则运⾏下⼀句。
如: if exist c:\config.sys type c:\config.sys
表⽰如果存在c:\config.sys⽂件,则显⽰它的内容。
****** 注 ********
也可以使⽤以下的⽤法:
if exist command
device 是指DOS系统中已加载的设备, 在win98下通常有:
AUX, PRN, CON, NUL
COM1, COM2, COM3, COM4
LPT1, LPT2, LPT3, LPT4
XMSXXXX0, EMMXXXX0
A: B: C: ...,
CLOCK$, CONFIG$, DblBuff$, IFS$HLP$
具体的内容会因硬软件环境的不同⽽略有差异, 使⽤这些设备名称时, 需要保证以下三点:
1. 该设备确实存在(由软件虚拟的设备除外)
2. 该设备驱动程序已加载(aux, prn等标准设备由系统缺省定义)
2. 该设备驱动程序已加载(aux, prn等标准设备由系统缺省定义)
3. 该设备已准备好(主要是指a: b: ..., , 等)
可通过命令 mem/d | find "device" /i 来检阅你的系统中所加载的设备
另外, 在DOS系统中, 设备也被认为是⼀种特殊的⽂件, ⽽⽂件也可以称作字符设备; 因为设备(device)与⽂件都是使⽤句柄(handle)来管理的, 句柄就是名字, 类似于⽂件名, 只不过句柄不是应⽤于磁盘管理, ⽽是应⽤于内存管理⽽已, 所谓设备加载也即指在内存中为其分配可引⽤的句柄.
==================================
3、if errorlevel <;数字> 待执⾏的命令
很多DOS程序在运⾏结束后会返回⼀个数字值⽤来表⽰程序运⾏的结果(或者状态),通过if errorlevel命令可以判断程序的返回值,根据不同的返回值来决定执⾏不同的命令(返回值必须按照从⼤到⼩的顺序排列)。如果返回值等于指定的数字,则条件成⽴,运⾏命令,否则运⾏下⼀句。
如if errorlevel 2 goto x2
==== 注 ===========
返回值从⼤到⼩的顺序排列不是必须的, ⽽只是执⾏命令为 goto 时的习惯⽤法, 当使⽤ set 作为执⾏命令时, 通常会从⼩到⼤顺序排列, ⽐如需将返回码置⼊环境变量, 就需使⽤以下的顺序形式:
if errorlevel 1 set el=1
if errorlevel 2 set el=2
if errorlevel 3 set el=3
if errorlevel 4 set el=4
if errorlevel 5 set el=5
...
当然, 也可以使⽤以下循环来替代, 原理是⼀致的:
for %%e in (1 2 3 4 5 6 ) do if errorlevel %%e set el=%%e
更⾼效简洁的⽤法, 可以参考我写的另⼀篇关于获取 errorlevel 的⽂章
出现此种现象的原因是, if errorlevel ⽐较返回码的判断条件并⾮等于, ⽽是⼤于等于. 由于 goto 的跳转特性, 由⼩到⼤排序会导致在较⼩的返回码处就跳出; ⽽由于 set命令的 "重复" 赋值特性, 由⼤到⼩排序会导致较⼩的返回码 "覆盖" 较⼤的返回码.
另外, 虽然 if errorlevel=<;数字> command 也是有效的命令⾏, 但也只是 command 解释命令⾏时将 = 作为命令⾏切分符⽽忽略掉罢了
===========================
⼆、goto 批处理⽂件运⾏到这⾥将跳到goto所指定的标号(标号即label,标号⽤:后跟标准字符串来定义)处,goto语句⼀般与if配合使⽤,根据不同的条件来执⾏不同的命令组。
如:
goto end
:end
echo this is the end
标号⽤":字符串"来定义,标号所在⾏不被执⾏。
==== willsort 编注
label 常被译为 "标签" , 但是这并不具有⼴泛的约定性.