相关文章推荐
气宇轩昂的蜡烛  ·  HTTP Redirects ...·  2 年前    · 
发财的杨桃  ·  rbenv + ruby-build: ...·  2 年前    · 
博学的墨镜  ·  npm - Cannot download ...·  2 年前    · 
批处理(Batch) ,也称为批处理 脚本 。顾名思义,批处理就是对某对象进行批量的处理,通常被认为是一种简化的 脚本语言 ,它应用于 DOS 和Windows系统中。 批处理文件 的扩展名为 bat 。比较常见的批处理包含两类: DOS 批处理和PS批处理。PS批处理是基于微软的强大的PowerShell的,用来 批量处理 一些任务的 脚本 ;而DOS批处理则是基于DOS命令的,用来自动地 批量 地执行DOS命令以实现特定操作的脚本。更复杂的情况,需要使用if、for、goto等命令控制程式的运行过程,如同 C 、Basic等高级语言一样。如果需要实现更复杂的应用,利用外部程式是必要的,这包括系统本身提供的外部命令和第三方提供的工具或者软件。 批处理程序 虽然是在命令行环境中运行,但不仅仅能使用命令行软件,任何当前系统下可运行的程序都可以放在批处理文件中运行。
有些人认为批处理语言的含义要比上面的描述更广泛,还包括许多软件自带的批处理语言,如 Microsoft Office Visual Studio Adobe Photoshop 所内置的批处理语言的功能,用户可通过它们让相应的软件执行自动化操作(例如调整某个资料夹所有 PSD 图档的解析度)。 而这类批处理语言也大多提供把一系列操作录制为批处理文件的功能,这样用户不必写程式就能得到批处理程序。
批处理文件的图标:旧(左),新(右) 批处理是一种简化的 脚本语言 ,也称作 。它应用于DOS和Windows系统中,它是由DOS或者Windows系统内嵌的命令 解释器 (通常是COMMAND. COM或者CMD.EXE)解释运行。类似于Unix中的 Shell脚本 。批处理文件具有.bat或者.cmd的扩展名,其最简单的例子,是逐行书写在命令行中会用到的各种命令。更复杂的情况,需要使用if,for,goto等命令控制程序的运行过程,如同C,Basic等中高级语言一样。如果需要实现更复杂的应用,利用外部程序是必要的,这包括系统本身提供的 外部命令 和第三方提供的工具或者 软件 。批处理文件,或称为 批处理程序 ,是由一条条的DOS命令组成的普通文本文件,可以用 记事本 直接编辑或用DOS命令创建,也可以用DOS下的 文本编辑器 Edit.exe来编辑。在“命令提示”下键入批处理文件的名称,或者双击该批处理文件,系统就会调用Cmd.exe运行该 批处理程序 。一般情况下,每条命令占据一行;当然也可以将多条命令用特定符号(如:&、&&、|、||等)分隔后写入同一行中;还有的情况就是像if、for等较高级的命令则要占据几行、几十甚至几百行的空间。系统在解释运行 批处理程序 时,首先扫描整个批处理程序,然后从第一行代码开始向下逐句执行所有的命令,直至程序结尾或遇见exit命令或出错意外退出。
以下这段代码是一个输出“hello world”的批处理代码:
@echo off
echo hello world
pause & exit
以下这段代码可以把用户输入的字符赋值到"input"变量里,并输出:
@echo off
set /p "input=>"
echo 您输入的是%input%
pause

批处理 echo 命令

打开 回显 或关闭请求回显功能,或显示消息。如果没有任何参数,echo命令将显示当前 回显 设置。
语法:
echo [{on|off}] [message]
示例:
@echo off
echo hello world
在实际应用中我们会把这条命令和重定向符号(也称为管道符号,一般用>,>>)结合来实现输入一些命令到特定的文件中,例如:
echo test>test.txt
命令会创建一个名为"test"的文本文件,并将"test"输入到文本文件中。
在批处理文件的开头,通常有:
@echo off
原因是"@"可以将本行的命令关闭回显,搭配"echo off"就可以不显示"echo off"的回显了。

批处理 rem 命令

注释命令,类似于在 C语言 中的/*--------*/,它并不会被执行,只是起一个注释的作用,只有在编辑批处理时才会被看到,主要用于方便修改。
"::" 也具有rem的功能
但"::"和rem还是有区别的,当关闭 回显 时,rem和::后的内容都不会显示。但是当打开 回显 时,rem后的内容会显示出来,然而"::"后的内容仍然不会显示。
@rem的效果跟"::"一样。
语法:
Rem [注释内容]
示例:
Rem 你好。
rem vs. ::
rem 123
:: 123
pause

批处理 pause 命令

暂停命令。运行 Pause 命令时,将显示下面的消息:
Press any key to continue. . .(或:请按 任意键 继续. . .)
示例:
@echo off
:begin
copy G:*.* d:\back
echo 请插入另一张光盘...
pause
goto begin
在这个例子中,驱动器 G 中磁盘上的所有文件均复制到d:\back中。显示的注释提示您将另一张 光盘 放入驱动器 G 时,pause 命令会使程序挂起,以便您更换 光盘 ,然后按 任意键 继续处理。
此命令还可以和echo命令配合使用达到自定义暂停的提示文本,如:
echo 请按下任意键来退出...
pause >nul
这段代码会输出:
请按下任意键来退出...

批处理 start 命令

调用外部程序,所有的DOS命令和命令行程序都可以由start命令来调用。
如:start calc.exe 即可打开Windows的计算器。
常用参数:
参数名
描述
MIN
开始时窗口最小化
SEPARATE
在分开的空间内开始 16 位 Windows 程序
HIGH
在 HIGH 优先级类别开始应用程序
REALTIME
在 REALTIME 优先级类别开始应用程序
WAIT
启动应用程序并等候它结束
parameters
这些为传送到命令/程序的参数
执行的 应用程序 是 32-位 GUI 应用程序时,CMD.EXE 不等应用程序终止就返回命令提示。如果在命令
脚本 内执行,该新行为则不会发生。

批处理 goto 命令

跳转命令。程序 指针 跳转到指定的标签,从标签后的第一条命令开始继续执行 批处理程序
语法:goto label (label是参数,指定所要转向的 批处理程序 中的行。)
示例:
:1
start
goto 1
运行以上这段命令,就会打开很多个窗口(用来整人相当不错)。这段命令的意思是,先用start命令打开窗口,再用goto 1命令转到1标签,即第一行后面的命令。注意标签名字前面要加上英文的冒号。
标签的名字可以随便起,但是最好是有意义的字母啦,字母前加个:用来表示这个字母是标签,goto
命令就是根据这个:来寻找下一步跳到哪里。最好有一些说明这样别人看起来才会理解你的意图啊。

批处理 set 命令

显示、设置或删除 变量
显示 变量 :set 或 set s 前者显示批处理当前已定义的所有变量及其值,后者显示所有以s开头的变量及值。
设置和调用 变量 :例如set aa=abcd(也可写作set "aa=abcd"),就是把aa定义为abcd。如果要调用这个 变量 ,就把aa两边加上个 百分号
示例:
set aa=abcd
echo %aa%
pause
运行这段命令就会显示“abcd”。
删除 变量 :"set aa="此句命令即可删除变量aa。若 变量 aa已被定义,则删除变量aa;若aa尚未定义,则此句命令无实质意义。
需要说明的是,批处理中的 变量 是不区分类型的,不需要像C语言中的变量那样还要区分int、float、char等。比如执行set aa=345后, 变量 aa的值既可以被视为数字345,也可以被视为字符串345。
set命令具有扩展功能,如用作交互输入、字符串处理、数值计算等,属于高级命令范畴。

批处理 重定向1 - ">" 与 ">>"

将输出信息重定向到指定的设备或文件。系统默认输出到显示器。
如:echo aaaaa>a.txt 即可将本在显示器上显示的信息aaaaa输出到文件a.txt中, 屏幕 上没有任何显示。如果文件a.txt本来已经存在,该命令将首先擦除a.txt中的所有信息,然后写入信息aaaaa;若a.txt本来就不存在,该命令即可新建一个a.txt文件,并写入信息aaaaa。
echo aaaaa>>a.txt 类似于echo aaaaa>a.txt。区别在于:如果a.txt本已存在,>a.txt会擦除a.txt中的原有内容,而>>a.txt并不擦除原有内容,仅在a.txt文件的末尾添加信息aaaaa。a.txt不存在时,二者没有差别。
所以,">"做的事情就是“覆盖”,而">>"做的事情就是“追加”。
需要注意的是,因为数字的特殊性,在echo输出数字时,请在后面加上空格,例如:echo 9313 >a.txt,否则有可能造成输出不完整。

批处理 管道符号 - "|"

将管道符号前面命令的输出结果重定向输出到管道符号后面的命令中去,作为后面命令的输入。
使用格式为:command_1|command_2
示例:
@echo off
echo aaaa>a.txt
del /p a.txt
pause
=========================================
@echo off
echo aaaa>a.txt
echo y|del /p a.txt
pause
对比以上两个批处理执行结果,读者即可明白管道符的用法和效果。
需要说明的是,上面del命令添加开关/p只是为了让读者明白管道符号的使用方法,实际 删除文件 时不加/p开关即可实现无提示直接删除。

批处理 转义符 - "^" 和 "%"

将特殊符号转化为一般符号,即剥离特殊符号的特殊地位。特殊符号指:"|"、"&"、">"。
比如,如果我们想输出符号“>”,直接用命令"echo >"是不行的,必须修改为"echo ^>"。其余几个特殊符号类似需要有同样的处理。
转义字符 使用举例:
@echo off
echo aaaa>a.txt
echo 第一句echo执行完毕
echo aaaa^>a.txt
echo 第二句echo执行完毕
pause
比较上面的两句echo,第一句echo将信息aaaa输出到了文件a.txt,而第二句echo则在直接屏幕上显示出aaaa>a.txt
除了"^" ,如果想要输出"%"该怎么办呢?在这里,"%"就可以当做"%"的转义符,例如:
echo 已完成100%%
这段代码就会输出:
已完成100%

批处理 逻辑命令

逻辑命令符包括:"&"、"&&"、"||"
"&"——它的作用是用来连接n个DOS命令,并把这些命令按 顺序执行 ,而不管是否有命令执行失败;
"&&"——当"&&";前面的命令成功执行时,执行"&&";后面的命令,否则不执行;
"||"——当"||"前面的命令失败时,执行"||"后面的命令,否则不执行。
@echo off
echo ^|^|
reg add HKCU /v try /f||echo **失败**
reg add HKCU1 /v try /f||echo **成功**
echo ^&^&
reg delete HKCU /v try /f&&echo **成功**
reg delete HKCU /v try /f&&echo **失败**
echo ^&
reg delete HKCU /v try /f&echo **也许成功**
reg delete HKCU /v try /f&echo **也许失败**
pause
执行reg add或reg delete后,系统会给出执行结果;我们通过echo命令也给出了“执行结果”。对比系统和我们自己给出的结果,既可以验证逻辑命令的判断机理。

批处理 if语句(选择结构)

if语句实现条件判断,包括字符串比较、存在判断、定义判断等。通过条件判断,if语句即可以实现选择功能。
1.字符串比较
if语句仅能够对两个字符(串)是否相同、先后顺序进行判断等。
其命令格式为:
IF [not] string1 compare-op string2 command1 [else command2]
其中,比较操作符compare-op有以下几类:
比较操作符
描述
==
等于
EQU
等于
NEQ
不等于
LSS
小于
LEQ
小于或等于
GTR
大于
GEQ
大于或等于
选择开关/i则不区分字符串大小写;选择not项,则对判断结果进行 逻辑非
字符串比较示例:
@echo off
set str1=abcd1233
set str2=ABCD1234
if %str1%==%str2% (echo 字符串相同!) else (echo 字符串不相同!)
if /i %str1% LSS %str2% (echo str1^=str2)
echo.
set /p choice=是否显示当前时间?(y/n)
if /i not %choice% EQU n echo 当前时间是:%date% %time%
pause>nul
对于最后一个if判断,当我们输入n或N时的效果是一样的,都不会显示时间。如果我们取消开关/i,则输入N时,依旧会显示时间。
另外请注意以下两个细节:
① echo str1^=str2 (对“=”转义)
② echo. (输出空行)
2.存在判断
存在判断的功能是判断文件或文件夹是否存在。
其命令格式为:
IF [NOT] EXIST filename command1 [else command2]
示例:
@echo off
if exist %0 echo 文件%0是存在的!
if not exist %~df0 (
  echo 文件夹%~df0不存在!
) else (echo 文件夹%~df0存在!)
pause>nul
这里注意几个地方:
①.存在判断既可以判断文件也可以判断文件夹;
②.%0即代表该批处理的全称(包括驱动器盘符、 路径 、文件名和扩展类型);
③.%~df0是对%0的修正,只保留了其驱动器盘符和路径,详情请参考for /?,属高级批处理范畴;
④.注意if语句的多行书写,多行书写要求command1的左括号必须和if在同一行、else必须和command1的右括号同行、command2的左括号必须与else同行、command1和command2都可以有任意多行,即command可以是命令集。
3.定义判断
定义判断的功能是判断 变量 是否存在,即是否已被定义。
其命令格式为:
IF [not] DEFINED variable command1 [else command2]
存在判断举例:
@echo off
set var=111
if defined var (echo var=%var%) else echo var尚未定义!
set var=% %
if defined var (echo var=%var%) else echo var尚未定义!
pause>nul
对比可知,"set var="可以取消 变量 ,收回变量所占据的内存空间。
4.结果判断
masm %1.asm
if errorlevel 1 pause & edit %1.asm
link %1.obj
先对 源代码 进行汇编,如果失败则暂停显示 错误信息 ,并在按 任意键 后自动进入编辑界面;否则用link 程序连接 生成的obj文件,这种用法是先判断前一个命令执行后的返回码(也叫 错误码 ,DOS程序在运行完后都有返回码),如果和定义的错误码符合(这里定义的错误码为1),则执行相应的操作(这里相应的操作为pause & edit %1.asm部分)。
另外,和其他两种用法一样,这种用法也可以表示否定。用否定的形式仍表达上面三句的意思,代码变为:
masm %1.asm
if not errorlevel 1 link %1.obj
pause & edit %1.asm

批处理 for语句(循环结构)

for语句可以实现类似于C语言里面的循环结构,当然for语句的功能要更强大一点,通过不同的开关可以实现更多的功能。for语句有多个开关,不同开关将会实现不同的功能。
1.无开关
无开关的for语句能够对设定的范围内进行循环,是最基本的for 循环语句 。其命令格式为:
FOR %%variable IN (set) DO command
其中,%%variable是 批处理程序 里面的书写格式,在DOS中书写为%variable,即只有一个 百分号 (%);set就是需要我们设定的循环范围,类似于C语言里面的循环 变量 ;do后面的command就是循环所执行的命令,即循环体。
无开关for语句举例:
@echo off
for %%i in (a,"b c",d) do echo %%i
pause>nul
2.开关/L
含开关/L的for语句,可以根据set里面的设置进行循环,从而实现对循环次数的直接控制。其命令格式为:
FOR /L %%variable IN (start,step,end) DO command
其中,start为开始计数的初始值,step为每次递增的值,end为结束值。当end小于start时,step需要设置为负数。
含开关/L的for语句举例(创建5个文件夹):
@echo off
for /l %%i in (1,2,10) do md %%i
pause
上例将新建5个文件夹,文件夹名称依次为1.3.5.7.9。可以发现,%%i的结束值并非end的值10,而是不大于(小于或等于)end的一个数。
3.开关/F
含开关/F的for语句具有最强大的功能,它能够对字符串进行操作,也能够对命令的返回值进行操作,还可以访问硬盘上的ASCII码文件,比如txt文档等。其命令格式为:
FOR /F ["options"] %%variable IN (set) DO command
其中,set为("string"、'command'、file-set)中的一个;options是(eol=c、skip=n、delims=xxx、tokens=x,y,m-n、usebackq)中的一个或多个的组合。各选项的意义参见下表《options选项详解》或者for /f。一般情况下,使用较多的是skip、tokens、delims三个选项。
options选项详解
eol=c
指定一个行注释符(只能是一个字符)。当脚本发现以注释符开头的行时,会自动忽略该行。
skip=n
从文件的开头开始,跳过n行。
delims=xx
指定一个分割符集(一个或多个字符)。默认分割符为空格和TAB
tokens=x,y,m-n
指定一个令牌或者令牌范围。
①当行被分割后,会分成几个部分,我们用令牌(tokens)来标记这些部分(令牌从1开始)。例如:有一行的内容为:aaaa bbbb cccc dddd,且分割符为空格,这一行就会被分成四个部分(即aaaa,bbbb,cccc,dddd),则令牌1就为aaaa,令牌2为bbbb,以此类推。
②tokens会造成额外的变量分配。
③”m-n”形式表示一个令牌范围。
④若字符串"tokens="的最后一个字符是通配符(*),则当最后一个标记解析完成后,一个额外变量会被分配,用来接收行内剩下的内容。
usebackq
作用于set:①.把单引号字符串当作普通的字符串;②.把双引号字符串当作文件 ③.把反引号(`)字符串当做命令
含开关/F的for语句举例:
@echo off
echo **No Options:
for /f %%a in ("1,2,10") do echo a=%%a
echo **Options tokens ^& delims:
for /f "tokens=1-3 delims=," %%a in ("1,2,10") do echo a=%%a b=%%b c=%%c
pause
注意第二个for循环中的“echo a=%%a b=%%b c=%%c”,%%a是在for语句中显式声明的,而%%b和%%c是因为“tokens=”选项被隐式声明的。它们按照26个字母的顺序被声明。如果在for语句中把%%a换成%%i,%%b和%%c应该替换成%%j和%%k。(i的后一个字母是j,j的后一个字母是k)
@echo off
echo 本文件夹里面的文件有:
dir>c:\file.txt
for /f "skip=5 tokens=3* delims= " %%a in (c:\file.txt) do (
  if not "%%a"=="<DIR>" if not "%%b"=="字节" if not "%%b"=="可用字节" echo %%b
del c:\file.txt
pause
对于后面的两个例子,其中options里面的delims= 是可以删除的,因为只要添加了/F开关系统就将delims的值默认为空格。
符号 字符 串中的最后一个字符星号,
那么额外的变量将在最后一个符号解析之后
分配并接受行的保留文本。本例中也可以改为4,不过文件名中有空格的文件,只能显示空格以前部分
同时我们也看到了,for语句的do后面的command也是可以分行的,只需要保证command的左括号和do在同一行就可以了。
4.开关/D或/R
含开关/D或/R的for语句是与目录或文件有关的命令,一般情况下很少使用。含开关/R的命令有时候被用于通过遍历文件夹来查找某一个文件或文件夹,故而列举此例。
含开关/R的for语句举例(文件夹遍历):
@echo off
setlocal enabledelayedexpansion
FOR /R d: %%i IN (.) DO (set dd=%%iset "dd=!dd:~0,-1!"echo !dd!)
pause
exit
上例即可以罗列出D盘下的所有文件夹,其速度要比命令"tree d:"慢多了,不过其返回结果的实用性则远远超过了tree命令。
一般情况下我们不推荐通过遍历文件夹来查找文件,特别是在查找某些程序(比如 QQ.exe )的位置时。推荐通过reg命令查找注册表来查找QQ的路径,以保证查找效率。
上例中也出现了几个新面孔,如setlocal、感叹号等。其中,感叹号其实就是变量 百分号 (%)的强化版。之所以要用!而不用%,是因为在for循环中,当一个变量被多次赋值时,%dd%所获取的仅仅是dd第一次被赋予的值;要想刷新dd的值,就必须首先通过命令"setlocal enabledelayedexpansion"来开启延迟变量开关,然后用!dd!来获取dd的值。
for语句是批处理里面功能最强大、使用最普遍却又最难掌握的一套命令,这也是批处理菜鸟和批处理高手最明显的一个分水岭,一旦掌握了这套命令,那么你就离批处理达人不远了!
截取字符串可以说是字符串处理功能中最常用的一个子功能了,能够实现截取字符串中的特定位置的一个或多个字符。举例说明其基本功能:
@echo off
set ifo=abcdefghijklmnopqrstuvwxyz0 12 3456789
echo 原字符串(第二行为各字符的序号):
echo %ifo%
echo 1234567890123 45678901234567890123456
echo 截取前5个字符:
echo %ifo:~0,5%
echo 截取最后5个字符:
echo %ifo:~-5%
echo 截取第一个到倒数第6个字符:
echo %ifo:~0,-5%
echo 从第4个字符开始,截取5个字符:
echo %ifo:~3,5%
echo 从倒数第14个字符开始,截取5个字符:
echo %ifo:~-14,5%
pause
当然,上面的例子只是将字符串处理的基本功能展示出来了,还看不出字符串处理具体有什么用处。下面这个例子是对时间进行处理。
@echo off
echo 当前时间是:%time% 即 %time:~0,2%点%time:~3,2%分%time:~6,2%秒%time:~9,2%厘秒
pause

批处理 扩充字符串

“扩充”这个词汇来自于 微软 自己的翻译,意思就是对表示文件 路径 的字符串进行特殊的处理,具体功能罗列如下:
修饰符
描述
%~I
删除任何引号("),扩充 %I
%~fI
将 %I 扩充到一个完全合格的路径名
%~dI
仅将 %I 扩充到一个驱动器号
%~pI
仅将 %I 扩充到一个路径
%~nI
仅将 %I 扩充到一个文件名
%~xI
仅将 %I 扩充到一个文件扩展名
%~sI
扩充的路径只含有短名
%~aI
将 %I 扩充到文件的文件属性
%~tI
将 %I 扩充到文件的日期/时间
%~zI
将 %I 扩充到文件的大小
%~$PATH:I
查找列在路径环境变量的目录,并将 %I 扩充到找到的第一个完全合格的名称。如果环境变量名未被定义,或者没有找到文件,此组合键会扩充到空字符串。
可以组合修饰符来得到多重结果:
%~dpI
仅将 %I 扩充到一个驱动器号和路径
%~nxI
仅将 %I 扩充到一个文件名和扩展名
%~fsI
仅将 %I 扩充到一个带有短名的完整路径名
%~dp$PATH:i
查找列在路径环境变量的目录,并将 %I 扩充到找到的第一个驱动器号和路径。
%~ftzaI
将 %I 扩充到类似输出线路的 DIR
以上内容引用于for /?帮助信息。其中的I代表变量I,不过需要说明的是,不是所有的变量都能够进行扩充的,有两个条件:1.该字符串代表一个文件路径;2.变量要用%x来表示,x可取a-z A-Z 0-9共62个字符中的任意一个。举例说明:
@echo off
echo 正在运行的这个批处理:
echo 完全路径:%0
echo 去掉引号:%~0
echo 所在分区:%~d0
echo 所处路径:%~p0
echo 文件名:%~n0
echo 扩展名:%~x0
echo文件属性:%~a0
echo 修改时间:%~t0
echo 文件大小:%~z0
pause
其中的%0是批处理里面的参数,代表当前运行的批处理的完全路径。类似的还有%1-%9,分别代表传递来的第1-9个参数。例子如下:
@echo off
set aa=C:\Windows\PPP\a.btx
call :deal aaa %aa% "c c" ddd eee
pause>nul
:deal
echo %%0 = %0
echo %%1 = %1
echo %%2 = %2
echo %%3 = %3
echo %%4 = %4
echo %%5 = %5
其中, 变量 aa在之前是不可以扩充的,通过call命令并将aa作为 参数传递 给子函数:deal,将aa变量转换成了变量%1,即符合%x格式,从而可以进行字符串扩充。
至于%x中x取a-z A-Z的形式,可以复习一下for语句,for语句里面的 变量 就是用%x来表示的,因而可以直接进行扩充。
批处理里面的数值计算功能较弱,只能够进行整型计算,忽略浮点数的小数部分;同时数值计算的范围也受限于系统位数,对于目前较为常见的32位机来说,数值计算能处理的数值范围为0x80000000h~0x7FFFFFFFh,即-2147483648~+2147483647。
数值计算需要使用set命令,具体格式为 “set /a expression” 。其中,expression代表计算 表达式 ,计算表达式跟C语言里面的表达式基本上完全一致。set支持的 运算符 也跟C语言里面的一样,只是没有了増一减一。set支持的 运算符 及优先级排序如下:
运算符及优先级
运算符
描述
()
分组
!,~,-
一元运算符(逻辑非、按位非、取负)
*,/,%
算数运算符(乘、除得商、除得余数,即取余)
+,-
算数运算符(加、减)
>
逻辑移位(左移一位、右移一位)
&
按位“与”
^
按位“异”
|
按位“或”
=,*=, /=, %=, +=, -=,&=,^=,|=,<<=,>>=
赋值
,(逗号)
表达式分隔符(set可一次处理多个表达式)
我们知道,批处理中取 变量 的值是需要用%或者!的,而在set /a 中,直接用变量名称即可取得变量的值。另外,set支持 八进制 (数字前缀0)、十进制(数字无前缀)和十六进制(数字前缀0x),且支持不同进制之间的计算,如set /a a=123+0123+0x123,计算及显示结果为十进制。

批处理 一、环境变量

所谓“环境变量”,实际上就是C语言里面的“变量”的意思。批处理的 变量 可以分为两类,由 系统定义 系统变量 和由用户根据需要自定义的用户变量。
批处理的一些变量是由 操作系统 事先定义好的,可以适用于任何批处理,我们称这些特殊的变量为“ 系统变量 ”。系统变量有很多个,包括硬件类、 操作系统 类、文件路径类、系统时间类等。要查看所有的系统变量,请新打开一个cmd窗口,输入set回车即可。对几个比较常用的 变量 解释如下:
ComputerName:计算机名,即右键 我的电脑 --属性--计算机名 选项卡中的“完整的计算机名称”。
ComSpec: cmd.exe 完整路径名
HomeDrive: 系统分区 盘符,一般都是C盘,故而HomeDrive=C:
Path: 可执行文件 默认搜索路径。这个东东非常重要!待会儿单独讲解…
ProgramFiles:就是系统的Program Files的路径啦,一般都是C:\Program Files,这就是安装 软件 时默认的安装路径了
Prompt:个性化设置cmd提示符的必备武器!不过,我没怎么用过~
SystemDrive:包含系统根目录的分区,其实就是HomeDrive了
SystemRoot:系统根目录路径,一般都是C:\WINDOWS
Temp:Tmp 文件、程序等可使用的临时目录,默认是C:\WINDOWS\Temp或Tmp。几乎所有的程序在运行时都会在这个目录里面“临时”写入文件。一般情况下,程序写入的 临时文件 都应该被该程序删除,可惜的是,大部分的程序都很健忘,导致这个文件夹占据的空间越来越大,自然也就使我们的系统增肥喽。所以,我们要把它修改到其他分区,并且时时的清理里面的 临时文件
UserName: 当前用户名 ,即所登陆的账户名
UserProfile:当前用户的配置目录,一般都是C:\Documents and Settings\%UserName%(Windows XP),C:\Users\%UserName%(Windows 10)。默认情况下,我们的 桌面 就是这个目录下面的“桌面”文件夹;我的文档就是这个目录下面的“My Documents”文件夹。所以啦,往 桌面 上或我的文档里面放东西就是放到这个文件夹下面了,也就是放到C盘了,重装系统时要覆盖C盘内容的,所以桌面上或我的文档里面的东西当然就会Gone with the Wind了~解决方法有两个,一是保持良好的习惯,不把重要文件放到这两个地方;二是,修改默认设置,将这两个文件夹都移到其他分区。
WinDir: 操作系统 路径,其实就是SystemRoot了

批处理 二、用户变量

编写 批处理程序 时,用户根据需要自己定义的变量称之为用户变量。用户变量类似于C语言里面的变量,仅仅在定义该变量的程序中有效。
用户变量由set命令定义,这是批处理中非常非常重要的一个操作,从而使set命令成为批处理里面使用频率最高的几个命令之一。关于set命令的使用,参考set /?,本教程也会在后面对其进行讲解。
变量 引用】
前面的几节课里面,我们已经看到了如何 引用变量 ,即直接用变量名操作变量,通过"%"或"!"来获取变量的值。其中,只有在for语句里面重复对同一 变量 多次赋值时才需要使用"!",并且在使用"!"调用变量时,要首先“启用延迟 环境变量 扩充”,启动命令为:SetLocal EnableDelayedExpansion。另外需要说明的是,“启用延迟 环境变量 扩充”后,所有的"!"都将被视为“取变量值”的特殊符号,即使用"^!"也不能输出符号"!"。若要输出"!",则需要“停用延迟 环境变量 扩充”,命令为: SetLocal DisableDelayedExpansion

批处理 三、参数

跟C语言类似,在调用函数或其他批处理时可能需要传递参数。批处理的 参数传递 分为直接和间接两种传递参数的方法。
【 1.直接传递 】
直接传递参数,即在使用call命令时,不使用任何参数,在子函数或子批处理里面直接对 主函数 (也称父批处理)里面的 变量 进行修改。这跟汇编语言里面的 参数传递 方式类似。
直接传递参数举例:
@echo off
setlocal enabledelayedexpansion
set var=aCdehiM,?mnrstW y
echo %var%
call :deal
setlocal disabledelayedexpansion
set var=%var:?=!%
echo %var%
pause>nul
:deal
set tm=!var!
set var=
for %%i in (6,3,11,11,16,15,1,4,11,5,12,13,9,0,12,7,15,14,5,10,2,16,18,8) do (
set var=!var!!tm:~%%i,1!
goto :eof
可以发现,当我们把 变量 var作为参数赋予子函数:deal后,子函数对var的值进行了修改;当子函数返回后, 主函数 里面的var的值就已经是子函数里面var被修改后的值了。
该例子中,使用了本节课前面讲到的setlocal enabledelayedexpansion和setlocal disabledelayedexpansion,前者保证了var在for循环里面能够根据我们的意愿进行处理,后者保证了能够正确输出符号"!"。另外例子中还使用了命令set,利用set对字符串进行了处理。还有一个地方使用了语句goto :eof,该语句相当于C语言里面的return或汇编语言里面的RET,即子程序返回命令。需要说明的是,当子函数本身就在批处理文件的末尾的话,我们是可以省略这句话的,比如将此例的goto :eof删除是不会产生任何影响的。
【 2.间接传递 】
间接传递参数,即在使用call命令时,在其后面添加参数,形如call {[:label][ChildBatch]} Parameter1 Parameter2 ... ParameterN。这跟C语言里面传递参数的格式类似。不同于C语言,批处理中的子函数不需要定义 形参 ,更不需要指定参数的个数。传递过来的参数,在子函数或子批处理里面是以%1~%9的形式表示的,即%1~%9分别表示传递过来的第1~9个参数。
@echo off
call :deal aaa bbb "c c" ddd eee
pause>nul
:deal
echo %%0 = %0
echo %%1 = %1
echo %%2 = %2
echo %%3 = %3
echo %%4 = %4
echo %%5 = %5
通过这个例子就可以清晰的看到%n参数表示法的用法。 参数列表 中包含空格的依旧要用双引号(")引起来;另外,也可以看到,%0已经变成了子函数的标号了,而不是父批处理的文件名全称。
【 3.区别 】
这两种 参数传递 方法本质上是没有区别的,形式上,直接传递直接对原 变量 进行操作,丢失了原变量的值;间接传递则通过%n对原变量进行了简单的备份,并且通用性更强,即不限定原变量的名称。另外,使用%n还有一个非常大的好处,就是可以通过%~*i来加强处理 变量 的能力。关于%~*i,详细内容参见for /?。
针对二者的差别,可以根据情况决定使用哪种传递方式:
--1.作为参数的 变量 名固定、且在子函数中不需要对其进行备份的情况下,使用直接传递法;
--2.若将子函数作为一个通用的 程序模块 ,以适应于对不同 变量 的处理,或者作为参数的变量不需要备份时,使用间接传递法。
具体使用哪种方法,还需根据实际情况或使用习惯进行选择。

批处理 四、返回值

有些命令在执行之后将会返回一定的 错误值 (errorlevel),可以通过errorlevel的值判断命令执行的状况。这点类似于C语言里面的exit(num),num就是错误代码。
获取返回值errorlevel的方法就是,在执行命令后,立马调用返回值errorlevel,如echo %errorlevel%或者if %errorlevel%==1等命令。
errorlevel举例:
@echo off
reg add HKCU /v try /f>nul
reg delete HKCU /v try /f
if errorlevel 0 (echo 删除成功!) else (echo 删除失败!)
reg delete HKCU /v try /f
if %errorlevel%==0 (echo 删除成功!) else (echo 删除失败!)
pause>nul
上面例子中,由于第一成功的删除了注册表,导致第二次因为找不到注册表而宣告失败。同时我们也看到了errorlevel的使用方法,即if errorlevel 0和if %errorlevel%==0是一样的。也许你注意到了,里面还有个笑脸呢~O(∩_∩)O哈哈~这就是ASCII码啦,后面跟你讲啊…
一般情况下,程序或命令成功执行时,返回的errorlevel是0,错误时返回1或更高的值。当然,有些命令是没有返回值的,这点需要注意。
嗯,有没有想起前面有个类似的东西啊?对了,那就是"|"和"&&"了,这两个符号就是根据errorlevel的值来进行逻辑判断的。
可别告诉我您不知道什么是批处理,当面对一大堆需要重复操作的数据时,往往让我们感到头大,这时候我们会想到批处理 ,那ArcGIS给大家提供了哪些批处理的方法呢,让我们拭目以待。
假设我们需要给道路建立缓冲区,设计到的工具为Buffer。
第一种情况,一个图层中不同要素建立不同大小的缓冲区
不同类型的道路,我们需要建立不同大小的缓冲区,比如,一级道路建立10米的缓冲区,二级道路建立15米的缓冲区,三级道建立20米的缓冲区。
步骤:
1、 首先应该保证你的道路数据里面有一个属性字段是用来存储Buffer宽度信息的。
2、 应用Buffer工具,在对应的参数位置选择相应字段即可。
第二种情况,不同的图层建立不同大小的缓冲区
假设还有其他不同的数据,不仅仅是道路,这些图层都需要建立缓冲区。
步骤:
1、 找到Buffer工具,右键,选择Batch,打开批处理面板。从该面板上我们可以发现,参数与我们打开Buffer的参数是一样的,这个时候是不是可以考虑在EXCEL中批量编辑好,然后复制过来呢
2、 在Excel中编辑需要的数据。
3、 将在Excel中编辑的数据复制到Buffer的批处理面板中来。此处需要注意,如果要复制10行数据,需要在批处理面板中先选中10行,右键单击,选择“paste”。
4、 点击OK即可。
如果想要每个图层中每种类型的数据的缓冲区宽度都不一样,该怎样处理的?这个留给您来尝试吧,原理很简单,就是找到参数位置,写上对应的字段即可。
第三种情况,Python实现批处理
如果您觉得上面的方法比较麻烦,而且要求也比较严格,您可以考虑应用 Python 来处理。
步骤:
1、 编写代码
import arcpy,os
inFCs = arcpy.GetParameterAsText(0)
outWS = arcpy.GetParameterAsText(1)
dist = arcpy.GetParameterAsText(2)
inFCs = inFCs.split(";")
for inFC in inFCs:
fileName =os.path.split(inFC)[1]
arcpy.Buffer_analysis(inFC,outWS + "\\" + fileName, str(dist) + "meter")
上面这段代码,要求用户自己设置输入数据、输出数据、缓冲区宽度(单位为米)。这里输出缓冲区数据的名称和输入数据的名称一样,如果不想要这样,可以进一步修改代码。
2、 接下来将该代码增加到ArcToolbox中,步骤比较简单,这里不再赘述。
需要注意的地方为最后的参数设置部分,”输入数据”的“MutiValue”属性设置为”Yes”。(此部分的参数设置较条条框框较多,需要与你的代码相互对照)
3、 完成后,工具将会添加到您自己的工具箱中。打开该工具,您会看到界面像普通工具的界面一样,输入数据可以设置很多个,但是这个工具所有图层的缓冲区大小必须一样。