windows batch

进了新公司,开始做运维。公司目前主要是用windows系统,而且保密性比较高,还不知道内网哪里可以搞到python或者perl的安装包,很多自动化脚本都是用windows batch写的,所以开始学习windows batch。唉,真是活到老学到老阿。

整理了一些batch脚本常用的命令:

set 变量名=变量值

设置变量。
被设定的变量以%变量名%引用

rem constants.bat
set DataPath=D:\OptoStats\Date

rem myBatch.bat
call constants.bat
echo %DataPath%

@命令

是加在其它命令行的最前面,表示运行时不显示命令行本身。

@echo off

echo off

echo状态设为off,将不显示命令行(如每行前的C:>等类似标志) 。

echo 字符串

echo 字符串 :将输入的字符串显示在cmd屏幕上。

rem

表示此命令后的字符为解释行,不执行,只是给自己今后查找用的。也可以起到 rem 的注释作用, 而且更简洁有效

call

语法:

call [ [Drive:] [Path] FileName [BatchParameters]][:label [arguments]]

参数: [Drive:][Path] FileName 指定要调用的批处理程序的位置和名称。Filename 参数必须是.bat 或 .cmd 扩展名的类型文件。BatchParameters 指定批处理程序所需的命令行信息(即参数项)。

调用另一个批处理程序,并且不终止父批处理程序(如果不用call而直接调用别的批处理文件,那么执行完那个批处理文件后将无法返回当前文件并执行当前文件的后续命令)。

call 命令接受用作跳转目标的标签。如果在脚本或批处理文件外使用call,它将不会在命令行起作用。

cscript

CScript.exe 是 Windows宿主的一个版本,可以用来从命令行运行脚本【2】。

内置环境变量

%cd%

当前的完整路径

%date%

显示当前系统的时间。
可以对其进行解析,比如在windowscmd命令行窗口执行date命令后这个环境变量的值为:


date_batch.png

那么如下的各个操作的意义如下:
%date:~0,4% 表示从左向右指针向右偏0位,然后从指针偏移到的位置开始提取4位字符,结果是2014(年的值)
%date:~5,2% 表示指针从左向右偏移5位,然后从偏移处开始提取2位字符,结果是03(月的值)
%date:~8,2% 表示指针从左向右偏移8位,然后从偏移处开始提取2位字符,结果是01(日的值)
%date:~5% 表示指针从左向右偏移5位,然后提取所有的值
%date:~-5% 表示指针反方向偏移,从最右端开始,偏移5位,然后从指针处提取左边的所有数值。

注意:“2014-03-01 星期六”是个字符串,在计算机里指针是从0开始计数的,所以这串字符的指针意义上的第5位是0,月份的0,取两位刚好是03.

过程控制语句

for

以前常觉得DOS的命令行功能太弱,无法象UNIX一样可以用命令行完成非常复杂的操作。实际上,当MS从WIN2K开始将命令行增强后,已经借鉴了相当多UNIX的优点,虽然还无法做到象UNIX那么灵活,但已可完成绝大多数的任务。而在这些增强中,最明显的,就是FOR命令。

基本应用

简单说,FOR是个循环,可以用你指定的循环范围生成一系列命令。最简单的例子,就是人工指定循环范围,然后对每个值执行指定的命令。例如,想快速报告每个硬盘分区的剩余空间:

for %a in (c: d: e: f do @dir %a\   find "bytes free" 

将输出:

8 Dir(s) 1,361,334,272 bytes free 
15 Dir(s) 8,505,581,568 bytes free 
12 Dir(s) 12,975,149,056 bytes free 
7 Dir(s) 11,658,854,400 bytes free 
高级应用

这些还不是FOR最强大的功能。我认为它最强大的功能,表现在以下这些高级应用:

  1. 可以用 /r 参数遍历整个目录树
  2. 可以用 /f 参数将文本文件内容作为循环范围
  3. 可以用 /f 参数将某一命令执行结果作为循环范围
  4. 可以用 %~ 操作符将文件名分离成文件名、扩展名、盘符等独立部分
  • 用 /r 遍历目录树
    当用 . 或 *.txt 等文件名通配符作为 for /r 的循环范围时,可以对当前目录下所有文件(包括子目录里面的文件)进行操作。举个例子,你想在当前目录的所有txt文件(包括子目录)内容中查找"bluebear"字样,但由于find本身不能遍历子目录,所以我们用for:
for /r . %a in (*.txt) do @find "bluebear" %a 

find 前面的 @ 只是让输出结果不包括 find 命令本身。这是DOS很早就有的功能。和FOR无关。 当用 . 作为循环范围时,for 只将子目录的结构(目录名)作为循环范围,而不包括里面的文件。

  • 将某一文件内容或命令执行结果作为循环范围:
    假如你有一个文件 todel.txt,里面是所有要删除的文件列表,现在你想将里面列出的每个文件都删掉。假设这个文件是每个文件名占一行,象这样:
c:\temp\a1.txt 
c:\temp\a2.txt 
c:\temp\subdir\b3.txt 
c:\temp\subdir\b4.txt 

那么可以用FOR来完成:

for /f %a in (todel.txt) do del %a 

这个命令还可以更强大。比如你的 todel.txt 并不是象上面例子那么干净,而是由DIR直接生成,有一些没用的信息,比如这样:

Volume in drive D is DATA 
Volume Serial Number is C47C-9908 
Directory of D:\tmp 
09/26/2001 12:50 PM 18,426 alg0925.txt 
12/02/2001 04:29 AM 795 bsample.txt 
04/11/2002 04:18 AM 2,043 invitation.txt 
4 File(s) 25,651 bytes 
0 Dir(s) 4,060,700,672 bytes free 

for 仍然可以解出其中的文件名并进行操作:

for /f "skip=5 tokens=5" %a in (todel.txt) do @if exist %a echo %a 

输出为:

alg0925.txt 
bsample.txt 
invitation.txt 

skip=5表示跳过前5行(就是DIR输出的头部信息),tokens=5表示将每行的第5列作为循环值放入%a,正好是文件名。在这里加了一个文件存在判断,是因为最后一行的"free"刚好也是第5列,目前还想不出好的办法来滤掉最后两行,所以检查一下可保万无一失。

  • 可以用 /f 参数将某一命令执行结果作为循环范围
    非常有用的功能。比如,我们想知道目前的环境变量有哪些名字(我们只要名字,不要值)。可是SET命令的输出是“名字=值”的格式,现在可以用FOR来只取得名字部分:
FOR /F "delims==" %i IN ('set') DO @echo %i 

将看到:

ALLUSERSPROFILE 
APPDATA 
CLASSPATH 
CommonProgramFiles 
COMPUTERNAME 
ComSpec 
dircmd 
HOMEDRIVE 

这里是将set命令执行的结果拿来作为循环范围。delims==表示用=作为分隔符,由于FOR /F默认是用每行第一个TOKEN,所以可以分离出变量名。如果是想仅列出值:

FOR /F "delims== tokens=2" %i IN ('set') DO @echo %i 

tokens=2和前例相同,表示将第二列(由=作为分隔符)作为循环值。

再来个更有用的例子:
我们知道 date /t (/t表示不要询问用户输入)的输出是象这样的:

Sat 07/13/2002 

现在我想分离出日期部分,也就是13:

for /f "tokens=3 delims=/ " %a in ('date /t') do @echo %a 

实际上把 tokens后面换成1,2,3或4,你将分别得到Sat, 07, 13和2002。注意delims=/后面还有个空格,表示/和空格都是分隔符。由于这个空格delims必须是/f选项的最后一项。
再灵活一点,象本文开头提到的,将日期用2002-07-13的格式输出:

for /f "tokens=2,3,4 delims=/ " %a in ('date /t') do @echo %c-%a-%b 

当tokens后跟多个值时,将分别映射到%a, %b, %c等。实际上跟你指定的变量有关,如果你指定的是 %i, 它们就会用%i, %j, %k等。
灵活应用这一点,几乎没有做不了的事。

  • 可以用 %~ 操作符将文件名分离成文件名、扩展名、盘符等独立部分
    这个比较简单,就是说将循环变量的值自动分离成只要文件名,只要扩展名,或只要盘符等等。
    例:要将 c:\mp3下所有mp3的歌名列出,如果用一般的 dir /b/s 或 for /r ,将会是这样:
g:\mp3\Archived\05-18-01-A\游鸿明-下沙\游鸿明-01 下沙.mp3 
g:\mp3\Archived\05-18-01-A\游鸿明-下沙\游鸿明-02 21个人.mp3 

如果我只要歌名(不要路径和".mp3"):

游鸿明-01 下沙 
游鸿明-02 21个人 

...... 

那么可以用FOR命令:

for /r g:\mp3 %a in (*.mp3) do @echo %~na 

凡是 %~ 开头的操作符,都是文件名的分离操作。具体请看 for /? 帮助。

两个百分号

%%是在批处理文件中变量符号,如果在CMD执行的话就应该是一个%。批处理文件在被解析的时候其中一个%被过滤掉,所以才用两个。

pushd

pushd命令:把后面参数指向的路径放置在虚拟堆栈中,并且换到此路径。注意,其实是有两个步骤的操作,第一,把路径放入虚拟堆栈,第二,将当前目录切换到路径。

pushd 绝对路径(也就是你要调用的那个批处理程序的目录)..
call xxx.bat
popd..

参考文章:
【1】Windows batch文件学习笔记
【2】cscript.exe
【3】windows批处理脚本bat命令解析【2】如何获取日期、时间
【4】for教程
【4】windows批处理脚本bat命令解析【3】PUSHD/POPD命令

你可能感兴趣的:(windows batch)