第一章 批处理基础

第一节 常用批处理内部命令简介

批处理定义:顾名思义,批处理文件是将一系列命令按一定的顺序集合为一个可执行的文本文件,其扩展名为 BAT 或者 CMD。这些命令统称批处理命令。

小知识:可以在键盘上按下 Ctrl+C 组合键来强行终止一个批处理的执行过程。了解了大概意思后,我们正式开始学习.先看一个简单的例子!

@echo off
echo "欢迎来到非常 BAT!"
pause

​把上面的 3 条命令保存为文件 test.bat 或者 test.cmd 然后执行,他就会在屏幕上显示以下二行话:

欢迎来到非常 BAT!
请按任意键继续. . .

这就是一个简单批处理文件了,这个批处理文件一共就用了 2 条命令 "echo" 和"pause" 还有一个特殊符号"@"从上面这个简单的批处理中,我们可以发现其实批处理就是运用一些含有特殊意义的符号和一些完成指定功能的命令组合而成,那么在批处理中有多少这样的特殊符号和功能命令呢?我们现在就来仔细了解一下一些最常用的!

批处理的常见命令(未列举的命令还比较多,请查阅帮助信息)

  1. REM 和 ::
  2. ECHO 和 @
  3. PAUSE
  4. ERRORLEVEL
  5. TITLE
  6. COLOR
  7. mode
  8. GOTO 和 :
  9. FIND
  10. START
  11. assoc 和 ftype
  12. pushd 和 popd
  13. CALL
  14. shift
  15. IF
  16. setlocal 与 变量延迟

介绍命令

1、REM 和 ::

REM 为注释命令,一般用来给程序加上注解,该命令后的内容不被执行,但能回显。其次, :: 也可以起到 rem 的注释作用, 而且更简洁有效; 但有两点需要注意:

第一, 任何以冒号:开头的字符行, 在批处理中都被视作标号, 而直接忽略其后的所有内容。有效标号:冒号后紧跟一个以字母数字开头的字符串,goto 语句可以识别。无效标号:冒号后紧跟一个非字母数字的一个特殊符号,goto 无法识别的标号,可以起到注释作用,所以 :: 常被用作注释符号,其实 :+ 也可起注释作用。

第二, 与 rem 不同的是, :: 后的字符行在执行时不会回显, 无论是否用 echo on 打开命令行回显状态, 因为命令解释器不认为他是一个有效的命令行, 就此点来看, rem 在某些场合下将比 :: 更为适用; 另外, rem 可以用于 config.sys 文件中。

2、ECHO 和 @

打开回显或关闭回显功能,或显示消息。如果没有任何参数, echo 命令将显示当前回显设置。@字符放在命令前将关闭该命令回显,无论此时 echo 是否为打开状态。

语法:

echo [{ on|off }] [message]

echo. 此用法将显示一空行,相当于回车,非常有用。

执行 echo off 将关闭回显,它后面的所有命令都不显示命令本身,只显示执行后的结果,除非执行 echo on 命令。

执行@echo off 不但关闭以后命令的回显,连 echo off 命令本身也不显示了。

通常以@echo off 作为批处理程序的首行。

一般用 ECHO MESSAGE 来显示一个特定的消息。

例:

@Echo off 
Echo hello
Pause

运行显示:hello

3、PAUSE

停止系统命令的执行并显示下面的内容。

例:

PAUSE

运行显示:

请按任意键继续. . .

要显示其他提示语,可以这样用:

Echo 其他提示语 & pause > nul

4、errorlevel

程序返回码

echo %errorlevel%

每个命令运行结束,可以用这个命令行格式查看返回码判断刚才的命令是否执行成功。默认值为 0,一般命令执行出错会设 errorlevel 为 1。

5、title

设置 cmd 窗口的标题,例如:

title 新标题

可以看到 cmd 窗口的标题栏变了。

6、COLOR

设置默认的控制台前景和背景颜色。语法为:

COLOR [attr]

attr 指定控制台输出的颜色属性。颜色属性由两个十六进制数字指定 -- 第一个为背景,第二个则为前景。每个数字可以为以下任何值之一:

0 = 黑色  

1 = 蓝色  

2 = 绿色  

3 = 湖蓝色

4 = 红色  

5 = 紫色  

6 = 黄色  

7 = 白色  

8 = 灰色

9 = 淡蓝色

A = 淡绿色

B = 淡浅绿色

C = 淡红色

D = 淡紫色

E = 淡黄色

F = 亮白色

如果没有给定任何参数,该命令会将颜色还原到 CMD.EXE 启动时的颜色。这个值来自当前控制台窗口、/T 开关或DefaultColor 注册表值。

如果用相同的前景和背景颜色来执行 COLOR 命令,COLOR 命令会将 ERRORLEVEL 设置为 1。例如:

COLOR fc" 在亮白色上产生亮红色

7、mode

配置系统设备。

串行口:     MODE COMm[:] [BAUD=b] [PARITY=p] [DATA=d] [STOP=s]

                         [to=on|off] [xon=on|off] [odsr=on|off]

                         [octs=on|off] [dtr=on|off|hs]

                         [rts=on|off|hs|tg] [idsr=on|off]

设备状态:   MODE [device] [/STATUS]

打印重定向: MODE LPTn[:]=COMm[:]

选定代码页: MODE CON[:] CP SELECT=yyy

代码页状态: MODE CON[:] CP [/STATUS]

显示模式:   MODE CON[:] [COLS=c] [LINES=n]

击键率:     MODE CON[:] [RATE=r DELAY=d]

例:

mode con cols=113 lines=15 & color 9f

此命令设置 DOS 窗口大小:15 行,113 列

8、GOTO 和 :

会点编程的朋友就会知道这GOTO是跳转的意思。在批处理中允许以“:XXX”来构建一个标号,然后用 GOTO XXX 跳转到标号:XXX 处,然后执行标号后的命令。

例:

if {%1}=={} goto noparms

if "%2"=="" goto noparms

标签的名字可以随便起,但是最好是有意义的字符串,前加个冒号用来表示这个字符串是标签,goto 命令就是根据这个冒号(:)来寻找下一步跳到到那里。

例:

@echo off

:start

set /a var+=1

echo %var%

if %var% leq 3 GOTO start

pause

运行显示:

1

2

3

4

9、find

在文件中搜索字符串。

FIND [/V] [/C] [/N] [/I] [/OFF[LINE]] "string" [[drive:][path]filename[ ...]]

/V 显示所有未包含指定字符串的行。

/C 仅显示包含字符串的行数。

/N 显示行号。

/I 搜索字符串时忽略大小写。

/OFF[LINE] 不要跳过具有脱机属性集的文件。

"string" 指定要搜索的文字串,

[drive:][path]filename

指定要搜索的文件。

如果没有指定路径,FIND 将搜索键入的或者由另一命令产生的文字。

Find 常和 type 命令结合使用

Type [drive:][path]filename | find "string" [>tmpfile] #挑选包含 string 的行

Type [drive:][path]filename | find /v "string" #剔除文件中包含 string 的行

Type [drive:][path]filename | find /c #显示文件行数

以上用法将去除 find 命令自带的提示语(文件名提示)

例:

@echo off

echo 111 >test.txt

echo 222 >>test.txt

find "111" test.txt

del test.txt

pause

运行显示如下:

---------- TEST.TXT

111

请按任意键继续. . .

例:

@echo off

echo 111 >test.txt

echo 222 >>test.txt

type test.txt|find "111"

del test.txt

pause

运行显示如下:

111

请按任意键继续. . .

10、start 命令

批处理中调用外部程序的命令(该外部程序在新窗口中运行,批处理程序继续往下执行,不理会外部程序的运行状况),如果直接运行外部程序则必须等外部程序完成后才继续执行剩下的指令。

例:start explorer d:\

调用图形界面打开 D 盘

11、assoc 和 ftype

文件关联

assoc 设置'文件扩展名'关联,关联到'文件类型'

ftype 设置'文件类型'关联,关联到'执行程序和参数'

当你双击一个.txt 文件时,windows 并不是根据.txt 直接判断用 notepad.exe 打开。而是先判断.txt 属于 txtfile '文件类型',再调用 txtfile 关联的命令行 txtfile=%SystemRoot%\system32\NOTEPAD.EXE %1。可以在"文件夹选项"→"文件类型"里修改这 2 种关联。

assoc         #显示所有'文件扩展名'关联

assoc .txt    #显示.txt 代表的'文件类型',结果显示 .txt=txtfile

assoc .doc    #显示.doc 代表的'文件类型',结果显示 .doc=Word.Document.8

assoc .exe    #显示.exe 代表的'文件类型',结果显示 .exe=exefile

ftype         #显示所有'文件类型'关联

ftype exefile #显示 exefile 类型关联的命令行,结果显示 exefile="%1" %*

assoc .txt=Word.Document.8

设置.txt 为 word 类型的文档,可以看到.txt 文件的图标都变了

assoc .txt=txtfile

恢复.txt 的正确关联

ftype exefile="%1" %*

恢复 exefile 的正确关联

如果该关联已经被破坏,可以运行command.com,再输入这条命令。

12、pushd 和 popd

切换当前目录

@echo off

c: & cd\ & md mp3  #在 C:\ 建立 mp3 文件夹

md d:\mp4          #在 D:\ 建立 mp4 文件夹

cd /d d:\mp4       #更改当前目录为 d:\mp4

pushd c:\mp3       #保存当前目录,并切换当前目录为 c:\mp3

popd               #恢复当前目录为刚才保存的 d:\mp4

一般用处不大,在当前目录名不确定时,会有点帮助。(dos 编程中很有用)

13、CALL

在批处理执行过程中调用另一个批处理,当另一个批处理执行完后,再继续执行原来的批处理。

CALL [drive:][path]filename [batch-parameters]

调用的其它批处理程序。filename 参数必须具有 .bat 或 .cmd 扩展名。

CALL :label arguments

调用本文件内命令段,相当于子程序。被调用的命令段以标签:label 开头,并且以命令 goto :eof 结尾。

另外,批脚本文本参数参照(%0、%1、等等)已如下改变:

批脚本里的 %* 指出所有的参数(如 %1 %2 %3 %4 %5 ...)

批参数(%n)的替代已被增强。您可以使用以下语法:

%~1  - 删除引号("),扩充 %1

%~f1 - 将 %1 扩充到一个完全合格的路径名

%~d1 - 仅将 %1 扩充到一个驱动器号

%~p1 - 仅将 %1 扩充到一个路径

%~n1 - 仅将 %1 扩充到一个文件名

%~x1 - 仅将 %1 扩充到一个文件扩展名

%~s1 - 扩充的路径指含有短名

%~a1 - 将 %1 扩充到文件属性

%~t1 - 将 %1 扩充到文件的日期/时间

%~z1 - 将 %1 扩充到文件的大小

%~$PATH: 1 - 查找列在 PATH 环境变量的目录,并将 %1扩充到找到的第一个完全合格的名称。如果环境变量名未被定义,或者没有找到文件,此组合键会扩充到空字符串

可以组合修定符来取得多重结果:

%~dp1 - 将 %1 扩展到驱动器号和路径

%~nx1 - 将 %1 扩展到文件名和扩展名

%~dp$PATH:1 - 在列在 PATH 环境变量中的目录里查找 %1,并扩展到找到的第一个文件的驱动器号和路径。

%~ftza1 - 将 %1 扩展到类似 DIR 的输出行。

在上面的例子中,%1 和 PATH 可以被其他有效数值替换。

%~ 语法被一个有效参数号码终止。%~ 修定符不能用在 %* 上。

注意:参数扩充时不理会参数所代表的文件是否真实存在,均以当前目录进行扩展。

要理解上面的知识,下面的例子很关键。例:

@echo off 
Echo 产生一个临时文件 > tmp.txt
Rem 下行先保存当前目录,再将 c:\windows 设为当前目录
pushd c:\windows
Call :sub tmp.txt
Rem 下行恢复前次的当前目录
Popd
Call :sub tmp.txt
pause
Del tmp.txt
:sub
Echo 删除引号: %~1
Echo 扩充到路径: %~f1
Echo 扩充到一个驱动器号: %~d1
Echo 扩充到一个路径: %~p1
Echo 扩充到一个文件名: %~n1
Echo 扩充到一个文件扩展名: %~x1
Echo 扩充的路径指含有短名: %~s1
Echo 扩充到文件属性: %~a1
Echo 扩充到文件的日期/时间: %~t1
Echo 扩充到文件的大小: %~z1
Echo 扩展到驱动器号和路径:%~dp1
Echo 扩展到文件名和扩展名:%~nx1
Echo 扩展到类似 DIR 的输出行:%~ftza1
Echo.
Goto :eof

14、shift

更改批处理文件中可替换参数的位置。

SHIFT [/n]

如果命令扩展名被启用,SHIFT 命令支持/n 命令行开关;该命令行开关告诉命令从第 n 个参数开始移位;n 介于零和八之间。例如:

SHIFT /2

会将 %3 移位到 %2,将 %4 移位到 %3,等等;并且不影响 %0 和 %1。

15、IF

IF 条件判断语句,语法格式如下:

IF [NOT] ERRORLEVEL number command

IF [NOT] string1==string2 command

IF [NOT] EXIST filename command

下面逐一介绍,更详细的分析请看后面章节。

(1) IF [NOT] ERRORLEVEL number command

IF ERRORLEVEL 这个句子必须放在某一个命令的后面,执行命令后由 IF ERRORLEVEL 来判断命令的返回值。

Number 的数字取值范围 0~255,判断时值的排列顺序应该由大到小。返回的值大于等于指定的值时,条件成立

例:

@echo off

dir c:

rem 退出代码为>=1 就跳至标题 1 处执行,>=0 就跳至标题 0 处执行

IF ERRORLEVEL 1 goto 1

IF ERRORLEVEL 0 goto 0

Rem 上面的两行不可交换位置,否则失败了也显示成功。

:0

echo 命令执行成功!

Rem 程序执行完毕跳至标题 exit 处退出

goto exit

:1

echo 命令执行失败!

Rem 程序执行完毕跳至标题 exit 处退出

goto exit

:exit

pause

运行显示:

命令执行成功!

(2) IF [NOT] string1==string2 command

string1 和 string2 都为字符的数据,比较区分大小写。条件相等后即执行后面的 command

检测当前变量的值做出判断,为了防止字符串中含有空格,可用以下格式

if [NOT] {string1}=={string2} command

if [NOT] [string1]==[string2] command

if [NOT] “string1”==“string2” command

这种写法实际上将括号或引号当成字符串的一部分了,只要等号左右两边一致就行了,比如

下面的写法就不行:

if {string1}==[string2] command

(3) IF [NOT] EXIST filename command

EXIST filename 为文件或目录存在的意思

echo off

IF EXIST autoexec.bat echo 文件存在!

IF not EXIST autoexec.bat echo 文件不存在!

这个批处理大家可以放在 C 盘和 D 盘分别执行,看看效果

16、setlocal 与 变量延迟

我们先引入一个例子。

例 1:

@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:

@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:

@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. || 组合命令
  11. "" 字符串界定符
  12. , 逗号
  13. ; 分号
  14. () 括号
  15. ! 感叹号
  16. 批处理中可能会见到的其它特殊标记符:
    • CR(0D) 命令行结束符
    • Escape(1B) ANSI 转义字符引导符
    • Space(20) 常用的参数界定符
    • Tab(09) ; = 不常用的参数界定符
    • + COPY 命令文件连接符
    • * ? 文件通配符
    • / 参数开关引导符
    • : 批处理标签引导符

1、@ 命令行回显屏蔽符

这个字符在批处理中的意思是关闭当前行的回显。我们从前面知道 ECHO OFF 可以关闭掉整个批处理命令的回显,但不能关掉 ECHO OFF 这个命令,现在我们在 ECHO OFF 这个命令前加个@,就可以达到所有命令均不回显的要求。

2、% 批处理变量引导符

这个百分号严格来说是算不上命令的,它只是批处理中的参数而已(多个%一起使用的情况除外,以后还将详细介绍)。

引用变量用%var%,调用程序外部参数用%1 至%9 等等

%0 %1 %2 %3 %4 %5 %6 %7 %8 %9 %*为命令行传递给批处理的参数

%0 批处理文件本身,包括完整的路径和扩展名

%1 第一个参数

%9 第九个参数

%* 从第一个参数开始的所有参数

参数%0 具有特殊的功能,可以调用批处理自身,以达到批处理本身循环的目的,也可以复制文件自身等等。

例:最简单的复制文件自身的方法

copy %0 d:\wind.bat

%0 这个有点特殊,有几层意思,先讲%1-%9 的意思.

%1 返回批处理的第一个参数

%2 返回批处理的第二个参数

%3-%9 依此推类

返回批处理参数?到底怎么个返回法?

我们看这个例子,把下面的代码保存为 test.BAT 然后放到 C 盘下

@echo off

echo %1 %2 %3 %4

echo %1

echo %2

echo %3

echo %4

进入 CMD,输入 cd c:\

然后输入 test.bat 我是第一个参数 我是第二个参数 我是第三个参数 我是第四个参数

注意中间的空格,我们会看到这样的结果:

我是第一个参数 我是第二个参数 我是第三个参数 我是第四个参数

我是第一个参数

我是第二个参数

我是第三个参数

我是第四个参数

对比下代码,%1 就是”我是第一个参数” %2 就是”我是第二个参数”

这些%1 和%9 可以让批处理也能带参数运行,大大提高批处理功能!

还有一个%* 他只是返回参数而已,不过他是一次返回全部参数的值,不用再一个个的输入%1 %2 来确定

例子

@echo off

echo %*

同样保存为 test.bat 放到 C 盘

进入 CMD,输入 cd c:\

然后输入 test.bat 我是第一个参数 我是第二个参数 我是第三个参数 我是第四个参数

可以看到他一次把全部参数都显示出来了。

好现在开始讲那个比较特殊的%0

%0 这个不是返回参数的值了,他有两层意思!

第一层意思:返回批处理所在绝对路径

例子:

@echo off

echo %0

pause

保存为 test.BAT 放在桌面运行,会显示如下结果

"C:\Documents and Settings\Administrator\桌面\test.bat"

他把当前批处理执行的所在路经打印出来了,这就是返回批处理所在绝对路径的意思

第二层意思:无限循环执行 BAT

例子:

@echo off

net user

%0

保存为 BAT 执行,他就会无限循环执行 net user 这条命令,直到你手动停止。其实%0 就是第一参数%1 前面那个参数,当然就是批处理文件名(包括路径)。

3、> 重定向符

输出重定向命令

这个字符的意思是传递并且覆盖,他所起的作用是将运行的结果传递到后面的范围(后边可以是文件,也可以是默认的系统控制台)。

在 NT 系列命令行中,重定向的作用范围由整个命令行转变为单个命令语句,受到了命令分隔符&,&&,||和语句块的制约限制。

比如:

使用命令:echo hello >1.txt 将建立文件 1.txt,内容为"hello "(注意行尾有一空格)

使用命令:echo hello>1.txt 将建立文件 1.txt,内容为"hello"(注意行尾没有空格)

4、>> 重定向符

输出重定向命令

这个符号的作用和>有点类似,但他们的区别是>>是传递并在文件的末尾追加,而>是覆盖。

用法同上

同样拿 1.txt 做例子

使用命令:

echo hello > 1.txt

echo world >>1.txt

这时候 1.txt 内容如下:

hello

world

5、<、>&、<& 重定向符

这三个命令也是管道命令,但它们一般不常用,你只需要知道一下就可以了,当然如果想仔细研究的话,可以自己查一下资料。(本人已查过,网上也查不到相关资料)

<,输入重定向命令,从文件中读入命令输入,而不是从键盘中读入。

@echo off

echo 2005-05-01>temp.txt

date <temp.txt

del temp.txt

这样就可以不等待输入直接修改当前日期

>&,将一个句柄的输出写入到另一个句柄的输入中。

<&,刚好和>&相反,从一个句柄读取输入并将其写入到另一个句柄输出中。

常用句柄:0、1、2,未定义句柄:3—9

1>nul 表示禁止输出正确的信息

2>nul 表示禁止输出错误信息。

其中的 1 与 2 都是代表某个数据流输入输出的地址(NT CMD 称之为句柄,MSDOS 称之为设备)。

句柄 0:标准输入 stdin,键盘输入

句柄 1:标准输出 stdout,输出到命令提示符窗口(console,代码为 CON)

句柄 2:标准错误 stderr,输出到命令提示符窗口(console,代码为 CON)

其中的 stdin 可被<重定向,stdout 可被>、>>重定向,而 stderr 在 DOS 下不可直接重定向,只有通过 ctty 或其它命令将系统控制权转交给其它设备的方式,来间接完成。

6、| 命令管道符

格式:第一条命令 | 第二条命令 [| 第三条命令...]

将第一条命令的结果作为第二条命令的参数来使用。 例如在不使 format 的自动格式化参数时,我是这样来自动格式化 A 盘的

echo y|format a: /s /q /v:system

用过 format 的都知道,再格盘时要输入 y 来确认是否格盘,这个命令前加上 echo y 并用|字符来将 echo y 的结果传给 format 命令,从而达到自动输入 y 的目的(这条命令有危害性,测试时请慎重)

7、^ 转义字符

^是对特殊符号<,>,&的前导字符,在命令中他将以上 3 个符号的特殊功能去掉,仅仅只把他们当成符号而不使用他们的特殊意义。

比如

echo test ^>1.txt

结果则是:test > 1.txt

他没有追加在 1.txt 里,只是显示了出来。

另外,此转义字符还可以用作续行符号。

举个简单的例子:

@echo off

echo 英雄^

是^

好^

男人

pause

不用多说,自己试一下就明白了。

&、&&、||为组合命令,顾名思义,就是可以把多个命令组合起来当一个命令来执行。这在批处理脚本里是允许的,而且用的非常广泛。因为批处理认行不认命令数目。

8、& 组合命令

语法:第一条命令 & 第二条命令 [& 第三条命令...]

命令按照从前往后的顺序依次执行命令,而不管每一条命令执行执行成功与否。

比如:

dir z:\ & dir y:\ & dir c:\

以上命令会连续显示 z,y,c 盘的内容,不理会该盘是否存在。

9、&& 组合命令

语法:第一条命令 && 第二条命令 [&& 第三条命令...]

这个命令和上边的类似按照从前往后的顺序依次执行命令,但当一条命令失败时,后边的命令也不会执行。

dir z:\ && dir y:\ && dir c:\

10、|| 组合命令

语法:第一条命令 || 第二条命令 [|| 第三条命令...]

这个命令和上边的类似按照从前往后的顺序依次执行命令,但当一条命令失败后才执行后面的命令,当碰到执行正确的命令后将不执行后面的命令,如果没有出现正确的命令则一直执行完所有命令。

提示:组合命令和重定向命令一起使用必须注意优先级。管道命令的优先级高于重定向命令,重定向命令的优先级高于组合命令。

问题:把 C 盘和 D 盘的文件和文件夹列出到 a.txt 文件中。你将如何来搞定这道题?

看例:

dir c:\ && dir d:\ > a.txt

仔细研究一下这句执行后的结果,看看是否能达到题目的要求!错了!这样执行后 a.txt 里只有 D 盘的信息!为什么?就因为这里&&命令和>命令不能同时出现一个句子里(批处理把一行看成一个句子)!组合命令&&的优先级没有重定向命令>的优先级高!所以这句在执行时将本行分成这两部分:dir c:\和 dir d:\ > a.txt,而并不是如你想的这两部分:dir c:\ && dir d:\ 和> a.txt。要使用组合命令&&达到题目的要求,必须得这么写:

dir c:\ > a.txt && dir d:\ >> a.txt

这样,依据优先级高低,DOS 将把这句话分成以下两部分:dir c:\ > a.txt 和 dir d:\ >> a.txt。

当然这里还可以利用&命令:

dir c:\ > a.txt & dir d:\ >> a.txt

11、"" 字符串界定符

双引号允许在字符串中包含空格,进入一个特殊目录可以用如下方法

cd "program files"

cd progra~1

cd pro*

以上三种方法都可以进入 program files 这个目录

12、, 逗号

逗号相当于空格,比如”dir,c:\”相当于””"

13、; 分号

分号,当命令相同时,可以将不同目标用;来隔离,但执行效果不变,如执行过程中发生错误,则只返回错误报告,但程序仍会执行。比如:

dir c:\;d:\;e:\

以上命令相当于

dir c:\

dir d:\

dir e:\

14、() 括号

小括号在批处理编程中有特殊的作用,左右括号必须成对使用,括号中可以包括多行命令,这些命令将被看成一个整体,视为一条命令行。

括号在 for 语句和 if 语句中常见,用来嵌套使用循环或条件语句,其实括号()也可以单独使用,请看例子。

例:

命令:echo 1 & echo 2 & echo 3

可以写成:

(

echo 1

echo 2

echo 3

)

上面两种写法效果一样,这两种写法都被视为是一条命令行。

注意:这种多条命令被视为一条命令行时,如果其中有变量,就涉及到变量延迟的问题。

15、! 感叹号

没啥说的,在变量延迟问题中,用来表示变量,即%var%应该表示为!var!,请看前面的 setlocal命令介绍。

第二章 for 命令详解

讲 FOR 之前,先告诉各位新手朋友,如果你有什么命令不懂,直接在 CMD 下面输入name /? 这样的格式来看系统给出的帮助文件,比如 for /? 就会把 FOR 命令的帮助全部显示出来!

FOR 这条命令基本上都被用来处理文本,我们这次除了要说处理文本的作用外还要讲他的其他一些好用的功能!

看看他的基本格式(这里我引用的是批处理文件中的格式,直接在命令行输入只需要一个%号)

FOR 参数 %%变量名 IN (相关文件或命令) DO 执行的命令

参数:FOR 有 4 个参数 /d /l /r /f 他们的作用我在下面用例子解释

%%变量名 :这个变量名可以是单个的小写 a-z 或者大写 A-Z,他们区分大小写哦~, FOR 会把每个读取到的值给他!

相关文件或命令:FOR 要读取其中的元素然后赋值给变量。

执行的命令:对每个变量的值要执行什么操作就写在这。

看不懂我的这些说明,可以在 CMD 输入 for /?看系统提供的帮助! 现在开始讲每个参数的意思。

一、参数 /d

FOR /D %variable IN (set) DO command [command-parameters]

如果集中包含通配符,则指定与目录名匹配,而不与文件名匹配。

如果 Set (也就是我上面写的 "相关文件或命令") 包含通配符(* 和 ?),将对与 Set 相匹配的每个目录(而不是指定目录中的文件组)执行指定的 Command。

这个参数主要用于目录搜索,不会搜索文件,看这样的例子

@echo off

for /d %i in (c:\*) do echo %i

pause

运行会把 C 盘根目录下的全部目录名字打印出来,而文件名字一个也不显示!

再看一个例子,比如我们要把当前路径下文件夹的名字只有 1~3 个字母的打印出来

@echo off

for /d %i in (???) do echo %i

pause

这样的话,如果你当前目录下有目录名字只有 1~3 个字母的,就会显示出来,没有就不显示了。这里解释下*号和?号的作用,*号表示匹配 N 个字符,而?号表示匹配一个字符。

@echo off

for /d %i in (window?) do @echo %i

pause

保存到 C 盘下执行,会显示什么呢?自己看吧! 显示:windows。/D 参数只能显示当前目录下的目录名字,这个大家要注意!

二、参数 /R

FOR /R [[drive:]path] %variable IN (set) DO command [command-parameters]

检查以 [drive:]path 为根的目录树,指向每个目录中的FOR 语句。如果在 /R 后没有指定目录,则使用当前目录。如果集仅为一个单点(.)字符,则枚举该目录树。

递归

上面我们知道,/D 只能显示当前路径下的目录名字,那么现在这个/R 也是和目录有关,他可以把当前或者你指定路径下的文件名字全部读取,注意是文件名字,有什么用看例子!

请注意 2 点:

1、set 中的文件名如果含有通配符(?或*),则列举/R 参数指定的目录及其下面的所有子目录中与 set 相符合的所有文件,没有相符文件的目录则不列举。

2、相反,如果 set 中为具体文件名,不含通配符,则枚举该目录树(即列举该目录及其下面的所有子目录),而不管 set 中的指定文件是否存在。这与前面所说的单点(.)枚举目录树是一个道理,单点代表当前目录,也可视为一个文件。

例:

@echo off

for /r c:\ %i in (*.exe) do echo %i

pause

咱们把这个 BAT 保存到 D 盘随便哪里然后执行,我会就会看到,他把 C 盘根目录,和每个目录的子目录下面全部的 EXE 文件都列出来了!!!!

例:

@echo off

for /r %i in (*.exe) do @echo %i

pause

参数不一样了吧!这个命令前面没加那个 C:\也就是搜索路径,这样他就会以当前目录为搜索路径,比如你这个 BAT 你把他放在 d:\test 目录下执行,那么他就会把 D:\test 目录和他下面的子目录的全部 EXE 文件列出来!!!

例:

@echo off

for /r c:\ %i in (boot.ini) do echo %i

pause

运行本例发现枚举了 c 盘所有目录,为了只列举 boot.ini 存在的目录,可改成下面这样:

@echo off

for /r c:\ %i in (boot.ini) do if exist %i echo %i

pause

用这条命令搜索文件真不错。。。。。。

三、参数 /L

FOR /L %variable IN (start,step,end) DO command [command-parameters]

该集合表示以增量形式从开始到结束的一个数字序列。因此,(1,1,5) 将产生序列 1 2 3 4 5,(5,-1,1) 将产生序列 (5 4 3 2 1)。

使用迭代变量设置起始值 (Start#),然后逐步执行一组范围的值,直到该值超过所设置的终止值 (End#)。/L 将通过对 Start# 与 End# 进行比较来执行迭代变量。如果 Start# 小于End#,就会执行该命令。如果迭代变量超过 End#,则命令解释程序退出此循环。还可以使用负的 Step# 以递减数值的方式逐步执行此范围内的值。例如,(1,1,5) 生成序列 1 2 3 4 5,而 (5,-1,1) 则生成序列 (5 4 3 2 1)。

举个例子:

@echo off

for /l %i in (1,1,5) do @echo %i

pause

保存执行看效果,他会打印从 1 2 3 4 5 这样 5 个数字。

(1,1,5)这个参数也就是表示从 1 开始每次加 1 直到 5 终止!

看这个例子:

@echo off

for /l %i in (1,1,5) do start cmd

pause

执行后是不是吓了一跳,怎么多了 5 个 CMD 窗口,呵呵!如果把那个 (1,1,5)改成 (1,1,65535)会有什么结果,我先告诉大家,会打开 65535 个 CMD 窗口....这么多你不死机算你强!

当然我们也可以把那个 start cmd 改成 md %i 这样就会建立指定个目录了!!!名字为 1-65535

看完这个被我赋予破坏性质的参数后,我们来看最后一个参数。

四、参数 /F

迭代及文件解析

使用文件解析来处理命令输出、字符串及文件内容。使用迭代变量定义要检查的内容或字符串,并使用各种 options 进一步修改解析方式。使用 options 指定哪些令牌应该作为迭代变量传递。请注意:在没有使用令牌选项时,/F 将只检查第一个令牌。

文件解析过程包括读取输出、字符串或文件内容,将其分成独立的文本行以及再将每行解析成零个或更多个令牌。然后通过设置为令牌的迭代变量值,调用 for 循环。默认情况下,/F传递每个文件每一行的第一个空白分隔符号。跳过空行。

详细的帮助格式为:

FOR /F ["options"] %variable IN (file-set) DO command [command-parameters]

FOR /F ["options"] %variable IN ("string") DO command [command-parameters]

FOR /F ["options"] %variable IN ('command') DO command [command-parameters]

带引号的字符串"options"包括一个或多个

指定不同解析选项的关键字。这些关键字为:

eol=c          - 指一个行注释字符的结尾(就一个)

skip=n         - 指在文件开始时忽略的行数。

delims=xxx     - 指分隔符集。这个替换了空格和跳格键的默认分隔符集。

tokens=x,y,m-n - 指每行的哪一个符号被传递到每个迭代的 for 本身。这会导致额外变量名称的分配。m-n格式为一个范围。通过 nth 符号指定 mth。如果符号字符串中的最后一个字符星号,那么额外的变量将在最后一个符号解析之后分配并接受行的保留文本。

usebackq       - 使用后引号(键盘上数字 1 左面的那个键`)。未使用参数 usebackq 时:file-set 表示文件,但不能含有空格双引号表示字符串,即"string"单引号表示执行命令,即'command'。使用参数 usebackq 时:file-set 和"file-set"都表示文件当文件路径或名称中有空格时,就可以用双引号括起来单引号表示字符串,即'string'后引号表示命令执行,即`command`以上是用 for /?命令获得的帮助信息,直接复制过来的。

For 命令例 1:****************************************

@echo off

rem 首先建立临时文件 test.txt

echo ;注释行,这是临时文件,用完删除 >test.txt

echo 11 段 12 段 13 段 14 段 15 段 16 段 >>test.txt

echo 21 段,22 段,23 段,24 段,25 段,26 段 >>test.txt

echo 31 段-32 段-33 段-34 段-35 段-36 段 >>test.txt

FOR /F "eol=; tokens=1,3,* delims=,- " %%i in (test.txt) do echo %%i %%j %%k

Pause

Del test.txt

运行显示结果:

11 段 13 段 14 段 15 段 16 段

21 段 23 段 24 段,25 段,26 段

31 段 33 段 34 段-35 段-36 段

请按任意键继续. . .

为什么会这样? 我来解释:

eol=; 分号开头的行为注释行

tokens=1,3* 将每行第 1 段,第 3 段和剩余字段分别赋予变量%%i,%%j,%%k

delims=,- (减号后有一空格)以逗号减号和空格为分隔符,空格必须放在最后

For 命令例 2:****************************************

@echo off

FOR /F "eol= delims=" %%i in (test.txt) do echo %%i

Pause

运行将显示 test.txt 全部内容,包括注释行,不解释了哈。

For 命令例 3:****************************************

另外/F 参数还可以以输出命令的结果看这个例子

@echo off

FOR /F "delims=" %%i in ('net user') do @echo %%i

pause

这样你本机全部帐号名字就出来了把扩号内的内容用两个单引号引起来就表示那个当命令执行,FOR 会返回命令的每行结果,加那个"delims=" 是为了让含有空格的行能整行显示出来,不加就只显示空格左边一列!

第三章 FOR 命令中的变量

FOR 命令中有一些变量,他们的用法许多新手朋友还不太了解,今天给大家讲解他们的用法!

先把 FOR 的变量全部列出来:

~I - 删除任何引号("),扩展 %I

%~fI - 将 %I 扩展到一个完全合格的路径名

%~dI - 仅将 %I 扩展到一个驱动器号

%~pI - 仅将 %I 扩展到一个路径

%~nI - 仅将 %I 扩展到一个文件名

%~xI - 仅将 %I 扩展到一个文件扩展名

%~sI - 扩展的路径只含有短名

%~aI - 将 %I 扩展到文件的文件属性

%~tI - 将 %I 扩展到文件的日期/时间

%~zI - 将 %I 扩展到文件的大小

%~$PATH:I - 查找列在路径环境变量的目录,并将 %I 扩展到找到的第一个完全合格的名称。如果环境变量名未被定义,或者没有找到文件,此组合键会扩展到空字符串

我们可以看到每行都有一个大写字母"I",这个 I 其实就是我们在 FOR 带入的变量,我们 FOR语句代入的变量名是什么,这里就写什么。

比如:FOR /F %%z IN ('set') DO @echo %%z

这里我们代入的变量名是 z 那么我们就要把那个 I 改成 z,例如%~fI 改为%~fz,至于前面的%~p 这样的内容就是语法了!

一、 ~I

删除任何引号("),扩展 %I

这个变量的作用就如他的说明,删除引号!

我们来看这个例子:

首先建立临时文件temp.txt,内容如下

"1111

"2222"

3333"

"4444"44

"55"55"55

可建立个 BAT 文件代码如下:

@echo off

echo ^"1111>temp.txt

echo "2222">>temp.txt

echo 3333^">>temp.txt

echo "4444"44>>temp.txt

echo ^"55"55"55>>temp.txt

rem 上面建立临时文件,注意不成对的引号要加转义字符^,重定向符号前不要留空格

FOR /F "delims=" %%i IN (temp.txt) DO echo %%~i

pause

del temp.txt

执行后,我们看 CMD 的回显如下:

1111 #字符串前的引号被删除了

2222 #字符串首尾的引号都被删除了

3333" #字符串前无引号,后面的引号保留

4444"44 #字符串前面的引号删除了,而中间的引号保留

55"55"55 #字符串前面的引号删除了,而中间的引号保留

请按任意键继续. . .

和之前 temp.txt 中的内容对比一下,我们会发现第 1、2、5 行的引号都消失了,这就是删除引号~i 的作用了!

删除引号规则如下:

1、若字符串首尾同时存在引号,则删除首尾的引号;

2、若字符串尾不存在引号,则删除字符串首的引号;

3、如果字符串中间存在引号,或者只在尾部存在引号,则不删除。

二、 %~fI

将 %I 扩展到一个完全合格的路径名

看例子:

把代码保存放在随便哪个地方,我这里就放桌面吧.

FOR /F "delims==" %i IN ('dir /b') DO @echo %~fi

pause

执行后显示内容如下

C:\Documents and Settings\Administrator\桌面\test.bat

C:\Documents and Settings\Administrator\桌面\test.vbs

当我把代码中的 %~fi 直接改成%i

FOR /F "delims==" %i IN ('dir /b') DO @echo %i

pause

执行后就会显示以下内容:

test.bat

test.vbs

通过对比,我们很容易就看出没有路径了,这就是"将 %I 扩展到一个完全合格的路径名"的作用。也就是如果%i变量的内容是一个文件名的话,他就会把这个文件所在的绝对路径打印出来,而不只单单打印一个文件名。

三、 %~dI

仅将 %I 扩展到一个驱动器号,看例子:

代码如下,我还是放到桌面执行!

FOR /F "delims==" %%i IN ('dir /b') DO @echo %%~di

pause

执行后我 CMD 里显示如下

C:

C:

我桌面就两个文件 test.bat,test.vbs,%%~di 作用是,如果变量%%i的内容是一个文件或者目录名,他就会把他这文件或者目录所在的盘符号打印出来!

四、 %~pI

仅将 %I 扩展到一个路径。这个用法和上面一样,他只打印路径不打印文件名字

FOR /F "delims==" %%i IN ('dir /b') DO @echo %%~pi

pause

五、 %~nI

仅将 %I 扩展到文件名,例如:

FOR /F "delims==" %%i IN ('dir /b') DO @echo %%~ni

pause

六、 %~xI

仅将 %I 扩展到一个文件扩展名,只打印文件的扩展名:

FOR /F "delims==" %%i IN ('dir /b') DO @echo %%~xi

pause

七、 %~sI

扩展的路径只含有短名,打印绝对短文件名(例如:program files的绝对短文件名为pro~1)

FOR /F "delims==" %%i IN ('dir /b') DO @echo %%~si

pause

八、 %~aI

将 %I 扩展到文件的文件属性,打印文件的属性

FOR /F "delims==" %%i IN ('dir /b') DO @echo %%~ai

pause

九、 %~tI

将 %I 扩展到文件的日期/时间,打印文件建立的日期

FOR /F "delims==" %%i IN ('dir /b') DO @echo %%~ti

pause

十、 %~zI

将 %I 扩展到文件的大小,打印文件的大小

FOR /F "delims==" %%i IN ('dir /b') DO @echo %%~zi

pause

上面例子中的"delims=="可以改为"delims=",即不要分隔符。

十一、 %~$PATH:I

查找列在路径环境变量的目录,并将 %I 扩展到找到的第一个完全合格的名称。如果环境变量名未被定义,或者没有找到文件,此组合键会扩展到空字符串。

@echo off

FOR /F "delims=" %%i IN (“notepad.exe”) DO echo %%~$PATH:i

pause

然后把这些代码保存为批处理,放在桌面。

上面代码显示结果为 C:\WINDOWS\system32\notepad.exe

他的意思就在 PATH 变量里指定的路径里搜索 notepad.exe 文件,如果有 notepad.exe 则会把他所在绝对路径打印出来,没有就打印一个错误!

好了,FOR 的的变量就介绍到这了!

第四章 批处理中的变量

批处理中的变量,我把他分为两类,分别为"系统变量"和"自定义变量"。我们现在来详解这两个变量!

一、系统变量

他们的值由系统将其根据事先定义的条件自动赋值,也就是这些变量系统已经给他们定义了值,不需要我们来给他赋值,我们只需要调用而以! 我把他们全部列出来!

%ALLUSERSPROFILE% 本地 返回“所有用户”配置文件的位置。
%APPDATA% 本地 返回默认情况下应用程序存储数据的位置。
%CD% 本地 返回当前目录字符串。
%CMDCMDLINE% 本地 返回用来启动当前的 Cmd.exe 的准确命令行。
%CMDEXTVERSION% 系统 返回当前的“命令处理程序扩展”的版本号。
%COMPUTERNAME% 系统 返回计算机的名称。
%COMSPEC% 系统 返回命令行解释器可执行程序的准确路径。
%DATE% 系统 返回当前日期。使用与 date /t 命令相同的格式。由 Cmd.exe 生成。有关date 命令的详细信息,请参阅 Date。
%ERRORLEVEL% 系统 返回上一条命令的错误代码。通常用非零值表示错误。
%HOMEDRIVE% 系统 返回连接到用户主目录的本地工作站驱动器号。基于主目录值而设置。用户主目录是在“本地用户和组”中指定的。
%HOMEPATH% 系统 返回用户主目录的完整路径。基于主目录值而设置。用户主目录是在“本地用户和组”中指定的。
%HOMESHARE% 系统 返回用户的共享主目录的网络路径。基于主目录值而设置。用户主目录是在“本地用户和组”中指定的。
%LOGONSERVER% 本地 返回验证当前登录会话的域控制器的名称。
%NUMBER_OF_PROCESSORS% 系统 指定安装在计算机上的处理器的数目。
%OS% 系统 返回操作系统名称。Windows 2000 显示其操作系统为 Windows_NT。
%PATH% 系统 指定可执行文件的搜索路径。
%PATHEXT% 系统 返回操作系统认为可执行的文件扩展名的列表。
%PROCESSOR_ARCHITECTURE% 系统 返回处理器的芯片体系结构。值:x86 或 IA64基于Itanium
%PROCESSOR_IDENTFIER% 系统 返回处理器说明。
%PROCESSOR_LEVEL% 系统 返回计算机上安装的处理器的型号。
%PROCESSOR_REVISION% 系统 返回处理器的版本号。
%PROMPT% 本地 返回当前解释程序的命令提示符设置。由 Cmd.exe 生成。
%RANDOM% 系统 返回 0 到 32767 之间的任意十进制数字。由 Cmd.exe 生成。
%SYSTEMDRIVE% 系统 返回包含 Windows server operating system 根目录(即系统根目录)的驱动器。
%SYSTEMROOT% 系统 返回 Windows server operating system 根目录的位置。
%TEMP%/%TMP% 系统/用户 返回对当前登录用户可用的应用程序所使用的默认临时目录。有些应用程序需要 TEMP,而其他应用程序则需要 TMP。
%TIME% 系统 返回当前时间。使用与 time /t 命令相同的格式。由 Cmd.exe 生成。有关time 命令的详细信息,请参阅 Time。
%USERDOMAIN% 本地 返回包含用户帐户的域的名称。
%USERNAME% 本地 返回当前登录的用户的名称。
%USERPROFILE% 本地 返回当前用户的配置文件的位置。
%WINDIR% 系统 返回操作系统目录的位置。

这么多系统变量,我们如何知道他的值是什么呢? 在 CMD 里输入 echo %WINDIR%,这样就能显示一个变量的值了!

举个实际例子,比如我们要复制文件到当前帐号的启动目录里就可以这样

copy d:\1.bat "%USERPROFILE%\「开始」菜单\程序\启动\"

%USERNAME% 本地 返回当前登录的用户的名称。 注意有空格的目录要用引号引起来。

另外还有一些系统变量代表一个意思或者一个操作! 他们分别是%0 %1 %2 %3 %4 %5 ......一直到%9 还有一个%* ,详见第一章第二节。

另外还有一些变量,他们也表示一些功能,FOR 命令中的那些就是,FOR 变量已经说过,就不讲了。

二、自定义变量

顾名思义,自定义变量就是由我们来给他赋予值的变量

要使用自定义变量就得使用 set 命令了,看例子.

@echo off

set var=我是值

echo %var%

pause

保存为 BAT 执行,我们会看到 CMD 里返回一个 "我是值"。var 为变量名,=号右变的是要给变量的值。

这就是最简单的一种设置变量的方法了。如果我们想让用户手工输入变量的值,而不是在代码里指定,可以用用 set 命令的/p 参数

例子:

@echo off

set /p var=请输入变量的值

echo %var%

pause

var 变量名 =号右边的是提示语,不是变量的值

变量的值由我们运行后自己用键盘输入!

第五章 set 命令详解

在上一章中我简单的介绍了一下 SET 设置自定义变量的作用,现在我来具体讲一下 set 的其他功能。

一、用 set 命令设置自定义变量

显示、设置或删除 cmd.exe 环境变量。

SET [variable=[string]]

variable 指定环境变量名。

string 指定要指派给变量的一系列字符串。

要显示当前环境变量,键入不带参数的 SET。

SET 命令不允许变量名含有等号。

例子:

@echo off

set var=我是值

echo %var%

pause

请看 “set var=我是值 “,这就是 BAT 直接在批处理中设置变量的方法。set 是命令,var 是变量名,等号右边的"我是值"是变量的值。我们要在批处理中引用这个变量就把 var 变量名用两个%(百分号)扩起来,如%var%。

SET 还可以提供一个交互界面,让用户自己输入变量的值,然后我们在来根据这个值来做相应操作,现在我就来说说 SET 的这种语法,只需要加一个"/P"参数就可以了!

SET /P variable=[promptString]

例子:

@echo off

set /p var=请输入变量的值:

echo 您输入了 %var% ~_~

pause

set /p 是命令语法 var 是变量名 =号右边的"请输入变量的值: ",这个是提示语,不是变量的值了!

运行后,我们在提示语后面直接输入 1,就会显示一行:您输入了 1 ~_~。

好了,先回顾到这,现在讲 SET 其他功能。

使用 set /?查看 SET 的帮助我们发现 SET 除了我上面讲的

SET [variable=[string]]

SET /P variable=[promptString]

这两种语法外,还有如下几种语法:

SET /A expression

环境变量替换已如下增强:

%PATH:str1=str2%

%PATH:~10,5%

%PATH:~-10%

%PATH:~0,-2%

这机种语法有什么用处呢?下面我们来一个个讲解他们!

二、用 set 命令进行整数运算

语法:SET /A expression

/A 命令行开关指定等号右边的字符串为被评估的数字表达式。该表达式评估器很简单并以递减的优先权顺序支持下列操作:

() 分组
! ~ 一元运算符
* / % 算术运算符
+ - 算术运算符
<< >> 二进制逻辑移位
& 二进制按位“与”
^ 二进制按位“异”
| 二进制按位“或”
= *= /= %= += = 算术赋值
&= ^= |= <<= >>= 二进制运算赋值
, 表达式分隔符

简单解释一下:

set 的/A 参数就是让 SET 可以支持数学符号进行加减等一些数学运算!

现在开始举例子介绍这些数学符号的用法:

例:

@echo off

set /p input=请输入计算表达式:

set /a var=%input%

echo 计算结果:%input%=%var%

pause

请看下面几个运算过程:

注意:DOS 的算术运算只能精确到 32 位整数,有效值范围是-2147483647 至 2147483647,超出该数值范围时计算出错。

请输入计算表达式:1+9+20+30-10

计算结果:1+9+20+30-10=50

请按任意键继续. . .

请输入计算表达式:10/3

计算结果:10/3=3 #DOS 计算精确到整数,小数舍了。

请按任意键继续. . .

请输入计算表达式:-100+62

计算结果:-100+62=-38

请按任意键继续. . .

请输入计算表达式:100%3 #求余数

计算结果:100%3=1

请按任意键继续. . .

请输入计算表达式:(25+75)*2/(15+5)

计算结果:(25+75)*2/(15+5)=10

请按任意键继续. . .

请输入计算表达式:1234567890*9876543210

无效数字。数字精确度限为 32 位。

计算结果:1234567890*9876543210=

请按任意键继续. . .

运行 set /a a=1+1,b=2+1,c=3+1 后会显示一个 4,但我们用echo %a% %b% %c%后看结果,会发现其他数学运算也有效果,这就是"逗"号的作用!

有时候我们需要直接在原变量进行加减操作就可以用这种语法

set /a var+=1

对应原始语法就是

set /a var = %var% + 1

都是一样的结果,在原变量的值上在进行数学运算,不过这样写简单一点

再来一个:

set /a var*=2

其他都这么用,只要帮助里有这个语法!

另外还有一些用逻辑或取余操作符,这些符号按照上面的使用方法会报错的。比如我们在 CMD 里输入

set /a var=1 & 1

他并不会显示为 1,而是报错,为什么?对于这样的"逻辑或取余操作符",我们需要把他们用双引号引起来,也可以用转义字

符^,看例子

set /a var= 1 "&" 1 这样结果就显示出来了,其他逻辑或取余操作符用法

set /a var= 1 "+" 1 异运算

set /a var= 1 "%" 1 取模运算

set /a var= 3 "<<" 2 左移位运算, 3 的二进制为 11,左移 2 位为 1100,换成十进制就是 12,自行验证

set /a var= 4 ">>" 2 右移位运算,4 的二进制为 100,右移动 2 位为 1,结果为 1

补充:凡是按位计算均需换算成二进制,下面行中的符号均针对二进制

这些符号也可以用&= ^= |= <<= >>= 这样的简单用法如下,注意引号。

set /a var"&=" 1

等于

set /a var = %var% "&" 1

思考题:求 2 的 n 次方

答案:

@echo off

set /p n=请输入 2 的几次方:

set /a num=1^<^<n

echo %num%

pause

三、用 set 命令进行字符串处理

1、字符串替换

好了,符号说到这,现在说%PATH:str1=str2%

上面语法的意思就是:将字符串变量%PATH%中的 str1 替换为 str2

这个是替换变量值的内容,看例子

@echo off 
set a= bbs. verybat. cn
echo 替换前的值: "%a%"
set var=%a: =%
echo 替换后的值: "%var%"
pause

 

运行显示:

替换前的值: " bbs. verybat. cn"

替换后的值: "bbs.verybat.cn"

对比一下,我们发现他把变量%a%的空格给替换掉了,从这个例子,我们就可以发现%PATH:str1=str2%这个操作就是把变量%PATH%的里的 str1 全部用 str2 替换。

这就是 set 的替换字符的很好的功能! 替换功能先讲到这,下面讲字符串截取功能。

2、字符串截取(切片)

**********************************************

截取功能统一语法格式为:%a:~[m[,n]]%

**********************************************

方括号表示可选,%为变量标识符,a 为变量名,不可少,冒号用于分隔变量名和说明部分,符号~可以简单理解为“偏移”即可,m 为偏移量(缺省为 0),n 为截取长度(缺省为全部)

%PATH:~10,5% 这个什么意思,看例子:

截取功能例子 1:

@echo off

set a=bbs.verybat.cn

set var=%a:~1,2%

echo %var%

pause

执行后,我们会发现只显示了"bs"两个字母。set 是命令,var 是变量值,a 要进行字符操作的变量,"1"是从变量"a"第几位开始显示,"2"表示显示几位。合起来就是把变量 a 的值从第 2 位(偏移量 1)开始,把 2 个字符赋予给变量 var。

其他两种语法

%PATH:~-10%

%PATH:~0,-2%

他们也是显示指定变量指定几位的值的意思

%PATH:~-10% 看例子

截取功能例子 2:

@echo off

set a=bbs.verybat.cn

set var=%a:~-3%

echo %var%

pause

运行结果:.cn

这个就是把变量 a 倒数 3 位的值给变量 VAR

当然我们也可以改成这样

截取功能例子 3:

@echo off

set a=bbs.verybat.cn

set var=%a:~3%

echo %var%

pause

运行显示:.verybat.cn

这个就是把变量 a 的从第 3 位开始后面全部的值给变量 VAR

%PATH:~0,-2% 例子

截取功能例子 4:

@echo off

set a=bbs.verybat.cn

set var=%a:~0,-3%

echo %var%

pause

执行后,我们发现显示的是"bbs.verybat",少了".cn"。从结果分析,很容易分析出,这是把变量 a 的值从 0 位开始,

到倒数第三位之间的值全部赋予给 var。

如果改成这样

截取功能例子 5:

@echo off

set a=bbs.verybat.cn

set var=%a:~2,-3%

echo %var%

pause

运行显示:s.verybat

那么他就是显示从第 3 位(偏移量 2)开始减去倒数三位字符的值,并赋予给变量 var

思考题目:输入任意字符串,求字符串的长度

参考答案:

@echo off

set /p str=请输入任意长度的字符串:

echo 你输入了字符串:"%str%"

if not defined str (pause & goto :eof)

set num=0

:len

set /a num+=1

set str=%str:~1%

if defined str goto len

echo 字符串长度为:%num%

pause

好了 set 的一些用法,就介绍到这了。

第六章 if 命令讲解

在 CMD 使用 IF /?打开 IF 的系统帮助(自己看我就不全部列出来了),我们会发现 IF 有 3 种基本的用法!

IF [NOT] ERRORLEVEL number command

IF [NOT] string1==string2 command

IF [NOT] EXIST filename command 

ERRORLEVEL number 如果最后运行的程序返回一个大于等于指定数字的整数,条件为 true。

string1==string2 如果指定的文字字符串匹配,指定条件为 true。

EXIST filename 如果指定的文件名存在,指定条件为 true。

command 如果符合条件,指定要执行的命令。

ELSE 子句必须在 IF 之后出现在同一行上。例如:

IF EXIST filename (

del filename

) ELSE (

echo filename missing

)

第一种用法:IF [NOT] ERRORLEVEL number command

这个用法的基本做用是判断上一条命令执行结果的代码,以决定下一个步骤.

一般上一条命令的执行结果代码只有两结果,"成功"用 0 表示 "失败"用 1 表示.

举个例子:

@echo off

net user

IF %ERRORLEVEL% == 0 echo net user 执行成功了!

pause

这是个简单判断上条命令是否执行成功。

IF %ERRORLEVEL% == 0 echo net user 执行成功了!

代码改写成:

IF ERRORLEVEL 0 echo net user 执行成功了!

用这种语法,不管你的上面的命令是否执行成功,他都会认为命令成功了。当使用 if errorlevel 0 的句式时,它的含义是:如果错误码的值大于或等于0的时候,将执行某个操作;当使用if %errorlevel%==0……的句式时,它的含义是:如果错误码的值等于0的时候,将执行某操作。

%ERRORLEVEL%这是个系统变量,返回上条命令的执行结果代码! "成功"用 0 表示 "失败"用 1 表示。 这只是一般的情况,实际上,errorlevel 返回值可以在 0~255 之间,比如,xcopy 默认的errorlevel 值就有 5 个,分别表示 5 种执行状态:

0 文件复制没有错误。

1 if errorlevel 2 echo。

2 用户按 CTRL+C 终止了 xcopy。

4 出现了初始化错误。没有足够的内存或磁盘空间,或命令行上输入了无效的驱动器名称或语法。

5 出现了磁盘写入错误。

要判断上面 xcopy 命令的 5 种退出情况,应写成:

if errorlevel 5 echo 出现了磁盘写入错误

if errorlevel 4 echo 出现了初始化错误

if errorlevel 2 echo 用户按 CTRL+C 终止了 xcopy

if errorlevel 1 echo if errorlevel 2 echo

if errorlevel 0 echo 文件复制没有错误。

才能正确执行。

补充完毕。

再举几个例子给新手理解

@echo off

net usertest

IF %ERRORLEVEL% == 1 echo net user 执行失败了!

pause

这个是判断上一条命令是否执行失败的

@echo off

set /p var=随便输入个命令:

%var%

if %ERRORLEVEL% == 0 goto yes

goto no

:yes

echo !var! 执行成功了

pause

exit

:no

echo 基本上执行失败了..

pause

这个是根据你输入的命令,自动判断是成功还是失败了!

再来一个简化版的

@echo off

set /p var=随便输入个命令:

%var%

if %ERRORLEVEL% == 0 (echo %var%执行成功了) ELSE echo %var%执行失败了!

pause

else 后面写上执行失败后的操作!

当然我门还可以把 if else 这样的语句分成几行写出来,使他看上去好看点...

@echo off

set /p var=随便输入个命令:

%var%

if %ERRORLEVEL% == 0 (

echo !var! 执行成功了

) ELSE (

echo 基本上执行失败了..

)

pause

第二种用法: IF [NOT] string1==string2 command

这个呢就是用来比较变量或者字符的值是不是相等的.

例子

@echo off

set /p var=请输入第一个比较字符:

set /p var2=请输入第二个比较字符:

if %var% == %var2% (echo 我们相等) ELSE echo 我们不相等

pause

上面这个例子可以判断你输入的值是不是相等,但是你如果输入相同的字符,但是如果其中一个后面打了一个空格,这个例子还是会认为相等,如何让有空格的输入不相等呢?我们在比较字符上加个双引号就可以了.

@echo off

set /p var=请输入第一个比较字符:

set /p var2=请输入第二个比较字符(多输入个空格试试):

if "%var%" == "%var2%" (echo 我们相等) ELSE echo 我们不相等

pause

第三种用法:IF [NOT] EXIST filename command

这个就是判断某个文件或者文件夹是否存在的语法

例子

@echo off

if exist "c:\test" (echo 存在文件) ELSE echo 不存在文件

pause

判断的文件路径加引号是为了防止路径有空格,如果路径有空格加个双引号就不会出现判断出错了!

另外我们看到每条 IF 用法后都有个[NOT]语句,这啥意思?其他加上他的话,就表示判断我们的条件不成立时,没加他是判断条件成立时,比如上面这个例子

@echo off

if not exist "c:\test" (echo 不存在文件) ELSE echo 存在文件

pause

第四种用法:IF 增强的用法

IF [/I] string1 compare-op string2 command

IF CMDEXTVERSION number command

IF DEFINED variable command

首先简单介绍一下后面两个用法。CMDEXTVERSION 条件的作用跟 ERRORLEVEL 的一样,除了它是在跟与命令扩展名有关联的内部版本号比较。第一个版本是 1。每次对命令扩展名有相当大的增强时,版本号会增加一个。命令扩展名被停用时,CMDEXTVERSION 条件不是真的。

DEFINED 条件的作用跟 EXISTS 的一样,除了它的操作对象是一个变量。如果已定义变量,返回的结果是 true。

IF [/I] string1 compare-op string2 command

这个语句在判断字符时不区分字符的大小写。

@echo off

if a == A (echo 我们相等) ELSE echo 我们不相等

pause

执行后会显示:我们不相等

@echo off

if /i a == A (echo 我们相等) ELSE echo 我们不相等

pause

加上/I 不区分大小写就相等了!

最后面还有一些用来判断数字的符号

EQU - 等于

NEQ - 不等于

LSS - 小于

LEQ - 小于或等于

GTR - 大于

GEQ - 大于或等于

我就举一个例子,大家都懂数学...不讲多了

@echo off

set /p var=请输入一个数字:

if %var% LEQ 4 (echo 我小于等于 4) ELSE echo 我不小于等于 4

pause

四、子程序

在批处理程序中可以调用外部可运行程序,比如 exe 程序,也可调用其他批处理程序,这些也可以看作子程序,但是不够方便,如果被调用的程序很多,就显得不够简明了,很繁琐。

在 windowsXP 中,批处理可以调用本程序中的一个程序段,相当于子程序,这些子程序一般放在主程序后面。

子程序调用格式:

CALL :label arguments

子程序语法:

:label

command1

command2

...

commandn

goto :eof

传至子程序的参数在 call 语句中指定,在子程序中用%1、%2 至%9 的形式调用,而子程序返回主程序的数据只需在调用结束后直接引用就可以了,当然也可以指定返回变量,请看下面的例子。

子程序例 1:

@echo off

call :sub return 你好

echo 子程序返回值:%return%

pause

:sub

set %1=%2

goto :eof

运行结果:你好

子程序例 2:设计一个求多个整数相加的子程序

@echo off

set sum=0

call :sub sum 10 20 35

echo 数据求和结果:%sum%

pause

:sub

rem 参数 1 为返回变量名称

set /a %1=%1+%2

shift /2

if not "%2"=="" goto sub

goto :eof

运行结果:65

九、时间延迟

什么是时间延迟?顾名思义,就是执行一条命令后延迟一段时间再进行下一条命令。

1、利用 ping 命令延时

例:

@echo off

echo 延时前!

ping /n 3 127.0.0.1 >nul

echo 延时后!

pause

解说:用到了 ping 命令的“/n”参数,表示要发送多少次请求到指定的 ip 。本例中要发送 3次请求到本机的 ip(127.0.0.1)。127.0.0.1 可简写为 127.1。“>nul”就是屏蔽掉 ping 命令所显示的内容。

2、利用 for 命令延时

例:

@echo off

echo 延时前!

for /l %%i in (1,1,5000) do echo %%i>nul

echo 延时后!

pause

解说:原理很简单,就是利用一个计次循环并屏蔽它所显示的内容来达到延时的目的。

路径中的通配符

? 代表0-9中的一个数字或26个字母中的一个字母

* 代表任意字符(包括零字符)