bat中for /f 如何截取任意行
2017-09-18 20:07 nigaopeng 阅读(6137) 评论(0) 编辑 收藏 举报一、概述
for命令开关有很多,/L,/F,/R。这里仅对含有/F的for进行分析,这个可能是最常用的,也是最强的命令,主要用来处理文件和一些命令的输出结果。
1.命令格式:
(1).FOR /F ["options"] %%i IN (file) DO command
(2).FOR /F ["options"] %%i IN ("string") DO command--注意双引号
(3).FOR /F ["options"] %%i IN ('command') DO command--注意单引号
2.参数说明:
file代表一个或多个文件,可以使用通配符。
string 代表字符串
command代表命令
["options"]选项有多个:
eol=c - 指定一个行注释字符,遇到c开头的行就忽略掉。
skip=n - 指定在文件开始时忽略的行数。
delims=xxx - 指定分隔符。默认是空格和TAB。
tokens=x,y,m-n - 指每行的哪一个符号被传递到每个迭代
的 for 本身。这会导致额外变量名称的分配。m-n
格式为一个范围。通过 nth 符号指定 mth。如果
符号字符串中的最后一个字符星号,
那么额外的变量将在最后一个符号解析之后
分配并接受行的保留文本。
usebackq - 1.把单引号字符串作为命令;2.允许中使用双引号扩起文件名称。
二、详细介绍
1.FOR /F %%i IN (file) DO command
file为文件名,按照官方的说法是,for会依次将file中的文件打开,并且在进行到下一个文件之前将每个文件读取到内存,按照每一行为一个元素,忽略空白的行。
假如文件d:\out.txt中有如下内容:
第1行第1列第1行第2列第1行第3列 第2行第1列第2行第2列第2行第3列 第3行第1列第3行第2列第3行第3列
怎么遍历文件中的内容呢?下面的语句可以:
for /f %%i in (d:\out.txt) do echo %%i
执行过程:for会先打开out.txt,然后读出out.txt里面的所有内容,把它作为一个集合,并且以每一行(文件中无分隔符情况下)作为一个元素,用%%i依次代替每个元素,然后执行do后面的命令。
for /f会默认以每一行(无分隔符)来作为一个元素,但是有分隔符怎么办?假如out.txt内容变成如下:
第1行第1列 第1行第2列 第1行第3列 第2行第1列 第2行第2列 第2行第3列 第3行第1列 第3行第2列 第3行第3列
那么for /f %%i in (d:\out.txt) do echo %%i就无法将所有的内容显示出来。显示结果如下:
第1行第1列 第2行第1列 第3行第1列
因为命令默认是以空格和TAB为分隔符,所以它遇到分割符之后,默认只取第一个,就将分割符后面的都丢掉了。
如果我们还想把每一行再分解更小的内容,该怎么办呢?for命令提供了更多的参数,它们就是:delims和tokens
delims :告诉for每一行应该拿什么作为分隔符,默认的分隔符是空格和tab键
比如,我们执行下面的命令:
for /f "delims= " %%i in (d:\out.txt) do echo %%i
显示的结果:
第1行第1列 第2行第1列 第3行第1列
这个命令和for /f %%i in (d:\out.txt) do echo %%i的效果是一样的。
如果我们想要每一行的后面的元素,那又如何呢?这个时候就可以利用tokens参数,它的作用就是当你通过delims将每一行分为更小的元素时,由tokens来控制要取哪一个或哪几个。
还是上面的例子,执行如下命令:
for /f "tokens=2 delims= " %%i in (d:\out.txt) do echo %%i
执行结果:
第1行第2列 第2行第2列 第3行第2列
如果要显示第二列和第三列,则换成tokens=2,3或tokens=2-3,全部显示则使用通配符tokens=*。
注意:如果显示多个元素(非*),比如这里的tokens=2-3,语句应该写成这样:
for /f "tokens=2,3 delims= " %%i in (d:\out.txt) do echo %%i %%j
怎么多出一个%%j?
这是因为你的tokens后面要取每一行的两列,用%%i来替换第二列,用%%j来替换第三列。
并且必须是按照英文字母顺序排列的,%%j不能换成%%k,因为i后面是j。
执行结果为:
第1行第2列 第1行第3列 第2行第2列 第2行第3列 第3行第2列 第3行第3列
又如下面语句:
for /f "tokens=2,* delims= " %%i in (d:\out.txt) do echo %%i %%j
它显示从第二列到最后的一列,执行结果为:
第1行第2列 第1行第3列 第2行第2列 第2行第3列 第3行第2列 第3行第3列
用%%i代替第二列,用%%j代替剩余的所有。
最后还有skip,eol和usebackq。
skip就是要忽略文件的前多少行,eol用来指定当一行以什么符号开始时,就忽略它。比如:
for /f "skip=2 tokens=*" %%i in (d:\out.txt) do echo %%i
结果为:
第3行第1列 第3行第2列 第3行第3列
用skip来告诉for跳过前两行。
再如,当out.txt内容变成:
#第1行第1列 第1行第2列 第1行第3列 #第2行第1列 第2行第2列 第2行第3列 第3行第1列 第3行第2列 第3行第3列
for /f "eol=# tokens=*" %%i in (d:\out.txt) do echo %%i
结果是:
第3行第1列 第3行第2列 第3行第3列
用eol来告诉for忽略以“.”开头的行。
usebackq就是反转的意思,他把字符串("out.txt")当做文件,或者把命令('command')当做字符串。
例如:
for /f "usebackq eol=# tokens=*" %%i in ("d:\out.txt") do echo %%i
结果是:
第3行第1列 第3行第2列 第3行第3列
如果不加usebackq,则"d:\out.txt"会被当成字符串。
又例如:
for /f "usebackq tokens=*" %%i in ('echo helloworld!') do echo %%i
结果:
echo helloworld!
,这里命令被当成了字符串。
如果没有usebackq
for /f "tokens=*" %%i in ('echo helloworld!') do echo %%i
结果:
helloworld!
2.FOR /F ["options"] %%i IN ("string") DO command
例子:
for /f "tokens=2,* delims= " %%i in ("this is an example!") do echo %%i %%j
结果:
is an example!
对于字符串的处理类似于文件的处理,此处不再赘述。
3.FOR /F ["options"] %%i IN ('command') DO command
该命令将command的执行结果作为集合,后面的处理类似文件处理,此处不再赘述。
方法1. 用find命令查找指定字符串
@Echo Off For /f "delims=" %%i in ('Type 1.txt^|Find "疑似"') do (Echo %%i) Pause
如果我根本不知道1.txt内容,只知道共四行,想单独显示第二行,该怎么写?
@Echo Off&Setlocal Enabledelayedexpansion For /f "delims=" %%i in (1.txt) do ( Set /a n+=1 If !n!==2 Echo %%i ) Pause
只显示第二行
本人较为愚钝,没看懂,但能实现,谢谢。若能写个分别将1.txt中的2 4 6行内容分别赋给变量a b c 就再追加20分。谢谢
For /f "delims=" %%i in (1.txt) do ( Set /a n+=1 If !n!==2 Set a=%%i If !n!==4 Set b=%%i If !n!==6 Set c=%%i ) Echo %a% %b% %c% Pause