批处理-for命令详解
⼤纲
⼀前⾔
⼆ for语句的基本⽤法
三 for /f (delims、tokens、skip、eol、userbackq、变量延迟)
四 for /r (递归遍历)
五 for /d (遍历⽬录)
六 for /l (计数循环)
⼀、前⾔
在批处理中,for是最为强⼤的命令语句,它的出现,使得解析⽂本内容、遍历⽂件路径、数值递增/递减等操作成为可能;配合if、call、goto等流程控制语句,更是可以实现脚本复杂的⾃动化、智能化操作;合理使⽤for语句,还能使代码⼤为简化,免除各位编写⼤量重复语句之苦。⽽能否熟练使⽤for语句,已经成
为衡量⼀个⼈批处理⽔平⾼低最主要的标准。
在这个系列教程中,我将通过实际应⽤中频繁出现的例⼦,带领⼤家步⼊for语句的神奇之门,⼀步步迈向for语句的魔幻殿堂,使得⼤家在实际的应⽤中,能独⽴写出简洁⾼效的代码,在批处理的世界⾥⾃由驰骋。
注意:以下的讲解,都是基于简体中⽂版Windows XP Pro SP3的操作系统环境。
⼆、for语句的基本⽤法
正如⾊彩缤纷的七彩光芒是由红绿蓝三原⾊构成的⼀样,最复杂的for语句,也有其基本形态,它的模样是这样的:
在cmd窗⼝中:
FOR %variable IN (set) DO command [command-parameters]
在批处理⽂件中:
FOR %%variable IN (set) DO command [command-parameters]
具体例⼦:
For %i in (1 2 3) do @echo %i
之所以要区分cmd窗⼝和批处理⽂件两种环境,是因为在这两种环境下,命令语句表现出来的⾏为虽然基本⼀样,但是在细节上还是稍有不同。
最明显的⼀个差异就是:在cmd窗⼝中,for之后的形式变量I必须使⽤单百分号引⽤,即%i;⽽在批处理⽂件中,引⽤形式变量i必须使⽤双百分号,即%%i。
为了⽅便起见,若不是特别强调,以下的讲解都以批处理⽂件环境为例。
我们先来看⼀下for语句的基本要素都有些什么:
1、for、in和do是for语句的关键字,它们三个缺⼀不可;
2、%%I是for语句中对形式变量的引⽤,就算它在do后的语句中没有参与语句的执⾏,也是必须出现的;
3、in之后,do之前的括号不能省略;
4、command1表⽰字符串或变量,command2表⽰字符串、变量或命令语句;
现在,你可能已经会写⼀个简单的for语句了,⽐如:
[code1]
@echo off
for %%I in (bbs.bathome) do echo %%I
pause
保存为批处理⽂件并执⾏,将会在弹出的批处理窗⼝中看到这样的信息:
bbs.bathome
请按任意键继续...
很快地,你会觉得这个for语句是如此的简单,简单到你丝毫感受不出它的强⼤:这个for语句,和我直接⽤echo语句没什么两样啊!
是的,演⽰代码永远都只是演⽰⽽已,就像⼤多数⾼级语⾔的教科书⼀样,在引导新⼿学习的时候,基本上都是千篇⼀律地告诉⼤家如何编写⼀个能显⽰hello world! 的窗⼝,从这些演⽰代码中,你看不到它们具有多少实⽤性,你只是感到有点好奇:咦,居然弹出了⼀个窗⼝?⽚刻之后,你就会觉得索然⽆味。
那好吧,为了让⼤家对for更加感兴趣,我们先来分析⼀下for语句的⼀些注意事项,之后,再让⼤家看看更为强⼤的for语句实例。
1、for语句的形式变量I,可以换成26个字母中的任意⼀个,这些字母会区分⼤⼩写,也就是说,%%I和%%i会被认为不是同⼀个变量;形式变量I还可以换成其他的字符,但是,为了不与批处理中的%0~%9这10个形式变量发⽣冲突,请不要随意把%%I替换为%%0 ~%%9中的任意⼀个;
2、in和do之间的command1表⽰的字符串或变量可以是⼀个,也可以是多个,每⼀个字符串或变量,我们称之为⼀个元素,每个元素之间,⽤空格键、跳格键、逗号、分号或等号分隔;
3、for语句依次提取command1中的每⼀个元素,把它的值赋予形式变量I,带到do后的command2中参与命令的执⾏;并且每次只提取⼀个元素,然后执⾏⼀次do后的命令语句,⽽⽆论这个元素是否被带到command2中参与了command2的运⾏;当执⾏完⼀次do后的语句之后,再提取command1中的下⼀个元素,再执⾏⼀次command2,如此循环,直到command1中的所有元素都已经被提取完毕,该for语句
才宣告执⾏结束;
其中,第3点是最为关键的,它描述了for语句的执⾏过程,是for语句的精髓所在,⼤家⼀定要牢记这⼀条,才能深刻理解更为复杂的for流程。
有了以上的基础,我们再来看⼀个例⼦,这个例⼦修改了[code1]的部分内容,结果将⼤不⼀样:
[code2]
@echo off
for %%I in (bbs,bathome,net) do echo %%I
pause
和[code1]的执⾏结果[result1]相⽐,[result2]发⽣了如下变化:
1、显⽰结果分成了3⾏(不算最后⼀⾏中⽂提⽰);
2、每⼀⾏都从逗号处被切分;
如果把 bbs.bathome 这个字符串中的点号换为空格、跳格或等号,执⾏结果将和example2的执⾏结果别⽆⼆致。
现在,我们来分析⼀下[code2]代码中for语句的执⾏过程:
⾸先,for语句以逗号为分隔符,把 bbs,bathome 这个字符串切分成三个元素:bbs、bathome和cn,由此决定了do后的语句将会被执⾏3次;
然后,第⼀次执⾏过程是这样的:先把bbs 这个字符串作为形式变量I的值,带⼊do后的语句中加以执⾏,也就是执⾏echo %%I 语句,此时的I值为bbs,因此,第⼀次执⾏的结果,将会在屏幕上显⽰bbs这个字符串;第⼆次执⾏和第⼀次执⾏的过程是⼀样的,只不过此时I的值已经被替换为command1中的第⼆个元素了,也就是bathome 这个字符串;如此循环,当第三次echo执⾏完毕之后,整条for语句才算执⾏完毕,此时,将执⾏下⼀条语句,也就是pause命令。
其实,这个例⼦只⽐上⼀个例⼦多了⼀点花样,有趣了那么⼀点点:⼀条for语句的执⾏结果居然被分成了3⾏!
为了让⼤家见识⼀下for的真正威⼒,本⼈绞尽脑汁,翻帖⽆数,不得要领,万般⽆奈之下,只好亮出了尘封在箱底多年的⼀段代码:检测当前硬盘都有哪些分区。
[code3]
@echo off
bbs论坛是什么set str=c d e f g h i j k l m n o p q r s t u v w x y z
echo 当前硬盘的分区有:
for %%i in (%str%) do if exist %%i: echo %%i:
pause
这段代码能检测硬盘都有哪些分区,包括U盘和移动硬盘的分区,但是,当光驱中有盘的时候,也会被列出来,这是本代码的⼀个缺憾,在以后的讲解中,我将向⼤家讲述如何消除这个瑕疵,敬请关注本系列的后续章节。
⾼级应⽤:
想知道当前⽬录下都有哪些⽂件吗?请⽤下⾯的代码:
@echo off
for %%i in (*.*) do echo "%%i"
pause
想列出当前⽬录下所有的⽂本⽂件吗?请⽤下⾯的代码
@echo off
for %%i in (*.txt) do echo "%%i"
pause
想列出只⽤两个字符作为⽂件名的⽂本⽂件吗?(注:实际上这个代码是输出少于或等于两个字符作为⽂件名的⽂本⽂件)请⽤下⾯的代码:
@echo off
for %%i in (??.txt) do echo "%%i"
pause
题外话:
1、列出当前⽬录下各种⽂件的⽅法,最简单的还是⽤dir命令,但是,从以上代码中,各位可以加深对for语句执⾏流程的理解(⽤到了通配符*和?);
2、注意:以上代码不能列出含有隐藏或系统属性的⽂件;(注:这⾥其实有⼀个很有趣的现象,windows中的系统⽂件⼀般具备两种属性——隐藏和系统;但是你如果测试的话就会发现,加上+s属性,但是不加+h的⽂件是可以被简单的for显⽰出来的。
例如:
@echo off
attrib +
For %%i in (*.txt) do Echo %%i
pause
这⾥的1.txt在结果中显⽰出来了。所以“以上代码不能列出含有隐藏或系统属性的⽂件”是不准确的,⽽因该说成“以上代码不能列出含有隐藏属性的⽂件”)
三、⽂本解析显神威:for /f ⽤法详解
前⾔
for /f 是个⼗分强⼤的家伙。
如果说,for语句是批处理中最强⼤的语句的话,那么,for /f 就是精华中的精华。
for /f 的强⼤,和它拥有众多的开关密切相关。因为开关众多,所以⽤法复杂,本章将分成若⼲⼩节,为⼤家逐⼀介绍强⼤的 for /f 语句。
(⼀)为解析⽂本⽽⽣:for /f 的基本⽤法
所有的对象,⽆论是⽂件、窗体、还是控件,在所有的⾮机器语⾔看来,⽆外乎都是形如"c:\"、"CWnd"之类的⽂本信息;⽽所有的对象,具体的如ini⽂件中的某条配置信息、注册表中的某个键值、数据库中的某条记录……都只有转化为具有⼀定格式的⽂本信息,⽅可被代码识别、操控。可以说,编程的很⼤⼀部分⼯作,都是在想⽅设法绞尽脑汁如何提取这些⽂本信息。
⽽提取⽂本信息,则是for /f的拿⼿好戏:读取⽂件内容;提取某⼏⾏字符;截取某个字符⽚段;对提取到的内容再切分、打乱、杂糅……只要你所能想到的花样,for /f 都会想⽅设法帮你办到,因为,for /f 就是被设计成专门⽤于解析⽂本的。
先来看个例⼦。
假如有个⽂本⽂件,内容如下:
[txt1]
论坛的⽬标是:不求最⼤,但求最好,做最实⽤的批处理论坛。
论坛地址:bbs.bathome。
这⾥是:新⼿晋级的福地,⾼⼿论剑的天堂。
那么,将如下代码保存为d,并放在同⼀⽬录下运⾏,将会在屏幕上原样显⽰的内容:
[code4]
@echo off
for /f %%i in () do echo %%i
pause
这段代码,主要是让你树⽴这样⼀种观念:读取⽂本⽂件的内容(注:改为“逐⾏分析⽂本⽂件的内容”,因为读取⽂本⽂件内容的⽅法命令有很多,⽐如重定向输⼊,⼜⽐如type/more/find/sort等命令),请使⽤ for /f 语句!
进阶话题:for /f 语句是把整个⼀次性显⽰出来的?
在这段代码中,虽然执⾏结果是把中的所有内容都显⽰出来了,貌似 for /f 语句是把整个⼀次性显⽰到屏幕上,实际上并⾮如此。
⽆论for语句做何种变化,它的执⾏过程仍然遵循基本的for流程:依次处理每个元素,直到所有的元素都被处理为⽌。只不过在for /f语句中,这⾥的元素是指⽂件中的每⼀⾏,也就是说,for /f 语句是以⾏为单位处理⽂本⽂件的。这是⼀条极为重要的规则,在上⼀章中也强调过它的重要性,希望在接下来的学习过程中,你能时刻牢记这⼀原则,那么,很多问题将会迎刃⽽解。以下是验证这⼀说法的演⽰代码(在[code4]的基础上添加了&pause语句):
[code5]
@echo off
for /f %%i in () do echo %%i&pause
pause
(⼆)切分字符串的利器:delims=
也许你对[code4]这段代码不屑⼀顾:不就是把的内容显⽰出来了么?好像⽤处不⼤啊。
好吧,我们来玩个魔术。
还是[txt1]这段⽂本,把[code4]改造⼀下:
[code6]
@echo off
for /f "delims=," %%i in () do echo %%i
pause
再次运⾏d,看到什么变化了吗?
[result2]
论坛的⽬标是:不求最⼤
论坛地址:bbs.bathome。
这⾥是:新⼿晋级的福地
请按任意键继续...
结果,你惊奇地发现,每⾏第⼀个逗号之后的所有内容都不见了(如果有不存在逗号的⾏,则保留原样),也就说,你成功地提取到了每⾏第⼀个逗号之前的所有内容!
试想⼀下,这段代码会有什么⽤呢?
如果别⼈给了你⼀个软件清单,每⾏都是"英⽂软件名(逗号)中⽂软件名"的格式,⽽你却只想保留英⽂名的时候,这段代码将是多么有⽤啊!再假设,有这么⼀个IP⽂件,第⼀列是数字格式的IP地址,第⼆列是具体的空间地址,列与列之间⽤逗号分隔,⽽你想提取其中数字格式的IP,呵呵,我不说你也知道该怎么办了吧?
要是⽂本内容不是以逗号分隔,⽽是以其他符号分隔,那么,把"delims=,"的逗号换成相应的符号就可以了。
在这⾥,我们引⼊了⼀个新的开关:"delims=,",它的含义是:以逗号作为被处理的字符串的分隔符号。
在批处理中,指定分隔符号的⽅法是:添加⼀个形如 "delims=符号列表" 的开关,这样,被处理的每⾏字符串都会被符号列表中罗列出来的符号切分开来。
需要注意的是:如果没有指定"delims=符号列表"这个开关,那么,for /f 语句默认以空格键或跳格键作为分隔符号。请把[txt1]中不同位置上的标点符号改为空格或跳格,再运⾏[code4]试试。
进阶话题:如果我要指定的符号不⽌⼀个,该怎么办?
在上⾯的讲解中,我提到了指定分隔符号的⽅法:添加⼀个形如"delims=符号列表"的开关。不知道你注意到没有,我的说法是"符号列表"⽽⾮"符号",这是⼤有讲究的,因为,你可以⼀次性指定多个分隔符号!
还是以[txt1]为例,把[code6]再改造⼀下
[code7]
@echo off
for /f "delims=.," %%i in () do echo %%i
pause
结果显⽰:
[result3]
论坛的⽬标是:不求最⼤
论坛地址:bbs
这⾥是:新⼿晋级的福地
请按任意键继续...
这样,第⼀个点号或第⼀个逗号之前的内容都被提取出来了。
[code7]的执⾏过程是:逐⾏读取中的内容,以点号和逗号切分每⼀⾏的内容(不存在点号和逗号的⾏,则不再切分,为了描述的⽅便,我们把被点号或逗号切分的⼀个⼀个的字符串⽚段,称之为节),然后,for /f 会提取第⼀节的内容作为最终结果,显⽰在屏幕上。需要注意的是,在这⾥,所有⾏
的字符串被切分成了两个以上的节,但是,[code7]的代码只会提取第⼀节字符串的内容,因为 for /f 语句默认只提取第⼀节的符串。
(三)定点提取:tokens=
上⼀节在讲解 delims= 的时候,我⼀再强调 for /f 默认只能提取到第⼀节的内容,现在我们来思考⼀个问题:如果我要提取的内容不在第⼀节上,那怎么办?
这回,就该轮到 tokens= 出马了。
tokens= 后⾯⼀般跟的是数字,如 tokens=2,也可以跟多个,但是每个数字之间⽤逗号分隔,如 tokens=3,5,8,它们的含义分别是:提取第2节字符串、提取第3、第5和第8节字符串。注意,这⾥所说的“节”,是由 delims= 这⼀开关划分的,它的内容并不是⼀成不变的。
下⾯来看⼀个例⼦:
[txt2]
尺有所短,⼨有所长,学好批处理没商量,考虑问题复杂化,解决问题简洁化。
对[txt2]这段⽂本,假设它们保存在⽂件中,如果我想提取“学好批处理没商量”这句话,该如何写代码呢?
我们稍微观察⼀下[txt2]就会发现,如果以逗号作为切分符号,就正好可以把“学好批处理没商量”化为单独的⼀“节”,结合上⼀节的讲解,我们知道,"delims=," 这个开关是不可缺少的,⽽要提取的内容在以逗号切分的第3节上,那么,tokens= 后⾯的数字就应该是3了,最终的代码如下:
[code8]
@echo off
for /f "delims=, tokens=3" %%i in () do echo %%i
pause
如果我们现在要提取的不只⼀个“节”,⽽是多个,那⼜怎么办呢?⽐如,要提取以逗号切分的第2节和第5节字符串,是写成这样吗?
[code9]
@echo off
for /f "delims=, tokens=2,5" %%i in () do echo %%i
pause
运⾏批处理后发现,执⾏结果只显⽰了第2节的内容。
原来,echo 后⾯的%%i 只接收到了tokens=2,5 中第⼀个数值2所代表的那个字符串,⽽第⼆个数值5所代表的字符串因为没有变量来接收,所以就⽆法在执⾏结果中显⽰出来了。
那么,要如何接收 tokens= 后⾯多个数值所指代的内容呢?
for /f 语句对这种情况做如下规定:
如果tokens= 后⾯指定了多个数字,如果形式变量为%%i,那么,第⼀个数字指代的内容⽤第⼀个形式变量%%i来接收,第⼆个数字指代的内容⽤第⼆个形式变量%%j来接收,第三个数字指代的内容⽤第三个形式变量%%k来接收……第N个数字指代的内容⽤第N个形式变量来接收,其中,形式变量遵循字母的排序,第N个形式变量具体是
什么符号,由第⼀个形式变量来决定:如果第⼀个形式变量是%%i,那么,第⼆个形式变量就是%%j;如果第⼀个形式变量⽤的是%%x,那么,第⼆个形式变量就是%%y。
现在回头去看[code9],你应该知道如何修改才能满⾜题⽬的要求了吧?修改结果如下:
[code10]
@echo off
for /f "delims=, tokens=2,5" %%i in () do echo %%i %%j
pause
如果有这样⼀个要求:显⽰[txt2]中的内容,但是逗号要替换成空格,如何编写代码?
结合上⾯所学的内容,稍加思索,你可能很快就得出了答案:
[code11]
@echo off
for /f "delims=, tokens=1,2,3,4,5" %%i in () do echo %%i %%j %%k %%l %%m
pause
写完之后,你可能意识到这样⼀个问题:假如要提取的“节”数不是5,⽽是10,或者20,或者更多,难道我也得从1写到10、20或者更多吗?有没有更简洁的写法呢?
答案是有的,那就是:如果要提取的内容是连续的多“节”的话,那么,连续的数字可以只写最⼩值和最⼤值,中间⽤短横连接起来即可,⽐如tokens=1,2,3,4,5 可以简写为tokens=1-5 。
还可以把这个表达式写得更复杂⼀点:tokens=1,2-5,tokens=1-3,4,5,tokens=1-4,5……怎么⽅便就怎么写吧。
⼤家可能还看到⼀种⽐较怪异的写法:
[code12]
@echo off
for /f "delims=, tokens=1,*" %%i in () do echo %%i %%j
pause
结果,第⼀个逗号不见了,取代它的是⼀个空格符号,其余部分保持不变。
其中奥妙就在这个星号上⾯。
tokens=后⾯所接的星号具备这样的功能:字符串从左往右被切分成紧跟在*之前的数值所表⽰的节数之后,字符串的其余部分保持不变,整体被*所表⽰的⼀个变量接收。理论讲解是⽐较枯燥的,特别是为了严密起见,还使⽤了很多限定性的修饰词,导致句⼦很长,增加了理解的难度,我们还是结合[code12]来讲解⼀下吧。
[txt2] 的内容被切分,切分符号为逗号,当切分完第⼀节之后,切分动作不再继续下去,因为tokens=1,* 中,星号前⾯紧跟的是数字1;第⼀节字符串被切分完之后,其余部分字符串不做任何切分,整体作为第⼆节字符串,这样,[txt2]就被切分成了两节,分别被变量%%i和变量%%j接收。
以上⼏种切分⽅式可以结合在⼀起使⽤。不知道下⾯这段代码的含义你是否看得懂,如果看不懂的话,那就运⾏⼀下代码,然后反复揣摩,你⼀定会更加深刻地理解本节所讲解的内容的:
[code13]
@echo off
for /f "delims=, tokens=1,3-4,*" %%i in () do echo %%i %%j %%k %%l
pause
(四)跳过⽆关内容,直奔主题:skip=n
很多时候,有⽤的信息并不是贯穿⽂本内容的始终,⽽是位于第N⾏之后的⾏内,为了提⾼⽂本处理的效率,或者不受多余信息的⼲扰,for /f 允许你跳过这些⽆⽤的⾏,直接从第N+1⾏开始处理,这个时候,就需要使⽤参数 skip=n,其中,n是⼀个正整数,表⽰要跳过的⾏数。例如:
[code14]
@echo off
for /f "skip=2" %%i in () do echo %%i
pause
这段代码将跳过头两⾏内容,从第3⾏起显⽰中的信息。
(五)忽略以指定字符打头的⾏:eol=
在cmd窗⼝中敲⼊:for /?,相关的解释为:
eol=c -指⼀个⾏注释字符的结尾(就⼀个)
FOR /F "eol=; tokens=2,3* delims=, " %i in () do @echo %i %j %k
会分析 中的每⼀⾏,忽略以分号打头的那些⾏……
第⼀条解释狗屁不通,颇为费解:⾏注释字符的结尾是什么意思?“(就⼀个)”怎么回事?结合第⼆条解释,才知道eol有忽略指定⾏的功能。但是,这两条解释是互相⽭盾
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。
发表评论