windows command(12):FOR


戴荃-悟空

FOR语句,循环控制,可以解析文件,大杀器。

基本语法

1
for {%% | %}variable in (set) do command [ CommandLineOptions]

其中,
%variable%%variable必须,%variable用来表示从命令提示符传递过来的参数,%%variable表示bat脚本自身的参数,参数大小写敏感。
(set)必须,指定一个或多个文件、文件夹、一定范围的数值、字符串,可使用通配符,括号不可少。
command必须,指定对(set)集合中每一个元素执行的命令。
CommandLineOptions可选,主要有/D-目录,/R-递归目录,/L-数值,/F-过滤。

无参数

for %%f in (*.doc *.txt) do type %%f
当前目录中的.doc或者.txt文件,依次代替%%f的位置,然后作为type命令的参数执行。

for %%f in (*.txt) do @echo %%f
效果类似于dir /B *.txt

参数/D 目录

1
FOR /D {%% | %}variable IN (set) DO command [command-parameters]

只能匹配当前目录,不匹配任何文件,也不会匹配下一级目录。
for %%f in (D:\*) do @echo %%f - D盘下所有的文件
for /D %%f in (D:\*) do @echo %%f - D盘下所有的目录
for /D %%i in (???) do echo %%i - 显示当前目录下名字只有1-3个字母的目录

参数/R 递归目录

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

此命令会搜索指定路径[[drive:]path]及所有子目录中与set相符合的所有文件,注意是指定路径及所有子目录。

  • set中的文件名如果含有通配符(?或*),则列举/R参数指定的目录及其下面的所用子目录中与set相符合的所有文件,无相符文件的目录则不列举。
  • 如果set中为具体文件名,不含通配符,则枚举该目录树(即列举该目录及其下面的所有子目录),并在每个目录后面加上具体的文件名,而不管set中的指定文件是否存在。
    for /r c:\ %%i in (*.exe) do echo %%i - 把C盘根目录,和每个目录的子目录下面全部的EXE文件都列出来了
    for /r c:\ %%i in (boot.ini) do echo %%i - 枚举了c盘所有目录,并在目录后面加上boot.ini,不管是否存在该文件
    for /r c:\ %%i in (boot.ini) do if exist %%i echo %%i - 很好的搜索命令,列举boot.ini存在的目录

参数/L 数列

1
FOR /L {%% | %}ariable IN (start#,step#,end#) DO command [command-parameters]

该集表示以增量形式从开始到结束的一个数字序列。因此,(1,1,5)将产生序列1 2 3 4 5,(5,-1,1)将产生序列(5 4 3 2 1).
for /l %%i in (1,2,10) do @echo %%i - 输出1 3 5 7 9
for /l %%i in (100,-20,1) do @echo %%i - 输出100,80,60,40,20
for /l %%i in (1,1,5) do start cmd - 打开5个CMD窗口
for /l %%i in (1,1,5) do md %%i - 建立从1~5共5个文件夹
for /l %%i in (1,1,5) do rd /q %%i - 删除从1~5共5个文件夹

参数/F 迭代文件解析

Use file parsing to process command output, strings and file content. Use iterative variables to define the content or strings you want to examine and use the various ParsingKeywords options to further modify the parsing. Use the ParsingKeywords token option to specify which tokens should be passed as iterator variables. Note that when used without the token option, /F will only examine the first token.
File parsing consists of reading the output, string or file content, breaking it up into individual lines of text and then parsing each line into zero or more tokens. The for loop is then called with the iterator variable value set to the token. By default, /F passes the first blank separated token from each line of each file. Blank lines are skipped. The different syntaxes are:

1
2
3
for /F ["ParsingKeywords"] {%% | %}variable in (filenameset) do command [CommandLineOptions]
for /F ["ParsingKeywords"] {%% | %}variable in ("LiteralString") do command [CommandLineOptions]
for /F ["ParsingKeywords"] {%% | %}variable in ('command') do command [CommandLineOptions]

The filenameset argument specifies one or more file names. Each file is opened, read and processed before going on to the next file in filenameset. To override the default parsing behavior, specify “ParsingKeywords”. This is a quoted string that contains one or more keywords to specify different parsing options.
If you use the usebackq option, use one of the following syntaxes:

1
2
3
for /F ["usebackqParsingKeywords"] {%% | %}variable in ("filenameset") do command [CommandLineOptions]
for /F ["usebackqParsingKeywords"] {%% | %}variable in ('LiteralString') do command [CommandLineOptions]
for /F ["usebackqParsingKeywords"] {%% | %}variable in (`command`) do command [CommandLineOptions]

when using usebackq option, be careful about the difference:
filenameset -> "filenameset"
"LiteralString" -> 'LiteralString'
'command' -> `command`

The following table lists the parsing keywords that you can use for ParsingKeywords.
eol=c - 指一个行注释字符的结尾(就一个)
skip=n - 指在文件开始时忽略的行数。
delims=xxx - 指分隔符集。这个替换了空格和制表符的默认分隔符集。
tokens=x,y,m-n - 指每行的哪一个符号被传递到每个迭代的 for 本身。这会导致额外变量名称的分配。m-n
格式为一个范围。通过 nth 符号指定 mth。如果符号字符串中的最后一个字符星号,那么额外的变量将在最后一个符号解析之后
分配并接受行的保留文本。
usebackq - 指定新语法已在下类情况中使用:
在作为命令执行一个后引号的字符串并且一个单引号字符为文字字符串命令并允许在 file-set
中使用双引号扩起文件名称。
eol=;表示忽略;开头的行;
skip=1表示忽略第一行;
delims=,表示用,`对字符串分隔;tokens=2,3*`将每行中的第二个和第三个符号传递给 for 程序体

%k takes the left content of string
1
2
for /F "eol=; tokens=2,3* delims=, " %%i in ("A1,A2,A3,A4,A5") do echo %%i %%j %%k
A2 A3 A4,A5
command
1
2
FOR /F "tokens=1* delims==" %%i IN ('set') DO @echo [%%i----%%j]
[OS----Windows_NT]
command with substitude
1
for /F "usebackq delims==" %%x in (`dir /b`) do @echo %%~fx

Reference:

help FOR
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
对一组文件中的每一个文件执行某个特定命令。
FOR %variable IN (set) DO command [command-parameters]
%variable 指定一个单一字母可替换的参数。
(set) 指定一个或一组文件。可以使用通配符。
command 指定对每个文件执行的命令。
command-parameters
为特定命令指定参数或命令行开关。
在批处理程序中使用 FOR 命令时,指定变量请使用 %%variable
而不要用 %variable。变量名称是区分大小写的,所以 %i 不同于 %I.
如果启用命令扩展,则会支持下列 FOR 命令的其他格式:
FOR /D %variable IN (set) DO command [command-parameters]
如果集中包含通配符,则指定与目录名匹配,而不与文件名匹配。
FOR /R [[drive:]path] %variable IN (set) DO command [command-parameters]
检查以 [drive:]path 为根的目录树,指向每个目录中的 FOR 语句。
如果在 /R 后没有指定目录规范,则使用当前目录。如果集仅为一个单点(.)字符,
则枚举该目录树。
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)
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]
或者,如果有 usebackq 选项:
FOR /F ["usebackqoptions"] %variable IN ("file-set") DO command [command-parameters]
FOR /F ["usebackqoptions"] %variable IN ('string') DO command [command-parameters]
FOR /F ["usebackqoptions"] %variable IN (`command`) DO command [command-parameters]
fileset 为一个或多个文件名。继续到 fileset 中的下一个文件之前,
每份文件都被打开、读取并经过处理。处理包括读取文件,将其分成一行行的文字,
然后将每行解析成零或更多的符号。然后用已找到的符号字符串变量值调用 For 循环 。
以默认方式,/F 通过每个文件的每一行中分开的第一个空白符号。跳过空白行。
你可通过指定可选 "options" 参数替代默认解析操作。这个带引号的字符串包括一个
或多个指定不同解析选项的关键字。这些关键字为:
eol=c - 指一个行注释字符的结尾(就一个)
skip=n - 指在文件开始时忽略的行数。
delims=xxx - 指分隔符集。这个替换了空格和制表符的
默认分隔符集。
tokens=x,y,m-n - 指每行的哪一个符号被传递到每个迭代
的 for 本身。这会导致额外变量名称的分配。m-n
格式为一个范围。通过 nth 符号指定 mth。如果
符号字符串中的最后一个字符星号,
那么额外的变量将在最后一个符号解析之后
分配并接受行的保留文本。
usebackq - 指定新语法已在下类情况中使用:
在作为命令执行一个后引号的字符串并且一个单
引号字符为文字字符串命令并允许在 file-set
中使用双引号扩起文件名称。
某些范例可能有助:
FOR /F "eol=; tokens=2,3* delims=, " %i in (myfile.txt) do @echo %i %j %k
会分析 myfile.txt 中的每一行,忽略以分号打头的那些行,将
每行中的第二个和第三个符号传递给 for 函数体,用逗号和/或
空格分隔符号。请注意,此 for 函数体的语句引用 %i 来
获得第二个符号,引用 %j 来获得第三个符号,引用 %k
来获得第三个符号后的所有剩余符号。对于带有空格的文件
名,你需要用双引号将文件名括起来。为了用这种方式来使
用双引号,还需要使用 usebackq 选项,否则,双引号会
被理解成是用作定义某个要分析的字符串的。
%i 在 for 语句中显式声明,%j 和 %k 是通过
tokens= 选项隐式声明的。可以通过 tokens= 一行
指定最多 26 个符号,只要不试图声明一个高于字母 "z" 或
"Z" 的变量。请记住,FOR 变量是单一字母、分大小写和全局的变量;
而且,不能同时使用超过 52 个。
还可以在相邻字符串上使用 FOR /F 分析逻辑,方法是,
用单引号将括号之间的 file-set 括起来。这样,该字符
串会被当作一个文件中的一个单一输入行进行解析。
最后,可以用 FOR /F 命令来分析命令的输出。方法是,将
括号之间的 file-set 变成一个反括字符串。该字符串会
被当作命令行,传递到一个子 CMD.EXE,其输出会被捕获到
内存中,并被当作文件分析。如以下例子所示:
FOR /F "usebackq delims==" %i IN (`set`) DO @echo %i
会枚举当前环境中的环境变量名称。
另外,FOR 变量参照的替换已被增强。你现在可以使用下列
选项语法:
%~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
在以上例子中,%I 和 PATH 可用其他有效数值代替。%~ 语法
用一个有效的 FOR 变量名终止。选取类似 %I 的大写变量名
比较易读,而且避免与不分大小写的组合键混淆。

for in MS doc
CMD批处理循环,太强大了