【自动化运维】dos ------ for命令学习笔记

批处理中的for循环是非常强大且使用的,让我们先来看看他的参数

C:\>for /?
对一组文件中的每一个文件执行某个特定命令。

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 ["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]

    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 的大写变量名
比较易读,而且避免与不分大小写的组合键混淆。

C:\>

总的来说用法为:

  1. 不带参数
  2. /D匹配目录
  3. /R匹配文件
  4. /L数字序列
  5. /F处理文本

让我来仔细讲解每种用法!!!

用法讲解

  1. 不带参数
FOR %variable IN (set) DO command [command-parameters]

  %variable  指定一个单一字母可替换的参数。
  (set)      指定一个或一组文件。可以使用通配符。
  command    指定对每个文件执行的命令。
  command-parameters
             为特定命令指定参数或命令行开关。

在批处理程序中使用 FOR 命令时,指定变量请使用 %%variable
而不要用 %variable。变量名称是区分大小写的,所以 %i 不同于 %I.

如果启用命令扩展,则会支持下列 FOR 命令的其他格式:

大概的意思是不带参数是会依次将(set)位置的内容依次赋值给%%variable位置的变量,然后在去执行 command 位置的的内容 ,command-parameters 的位置的内容是 command的位置执行的内容的参数。

这么说你可能没有听懂,我举个例子吧!!!

for %a in (我,的,小,可,爱) do @echo %a 

我们将这端代码复制到cmd中执行后的效果如下:

C:\>for %a in (我,的,小,可,爱) do @echo %a
我
的
小
可
爱

C:\>

这句代码的执行流程为下:
首先将括号中的内容(我,的,小,可,爱)依此赋值给变量a,然后执行命令@echo %a, 将变量%a的内容在屏幕上打印出来!
其中echo命令的作用是显示echo后面跟的内容,比如所说我想在屏幕上打印 “我是最棒的!"这句话,我就可以这么写

echo 我是最棒的!

执行的结果是

C:\>echo 我是最棒的!
我是最棒的!

%a是前面for循环赋值的变量 echo %a就相当于输出变量的内容
那么 @echo %a是啥意思呢?
@最用是隐藏后面使用的命令,不让命令在屏幕上显示出来,也就是说仅仅将变量a的内容在屏幕上打印处理,不将命令 “@echo %a”显示出来。我们看一下不在符号@的效果.

C:\>for %a in (我,的,小,可,爱) do echo %a

C:\>echo 我
我

C:\>echo 的
的

C:\>echo 小
小

C:\>echo 可
可

C:\>echo 爱
爱

C:\>

可以看到他将每次echo命令执行的过程都输出出来了!
这就是不带参数的for循环效果,总结为一句话:
将in后面括号中的内容依次赋值给%%variable位置的变量,并执行do语句后面的内容.

  1. /D匹配目录

实际上这个说的目录指的就是文件夹,也就是说 /D 参数只会主动匹配文件夹不会主动匹配文件,

FOR /D %variable IN (set) DO command [command-parameters]

    如果集中包含通配符,则指定与目录名匹配,而不与文件名匹配。

IN语句后面的括号中可以填入正则表达式,我这里介绍两个最常用的正则符号

* #星号匹配任意字符1次到多次
? #问号匹配任意字符0次到1次

比如说我想匹配出文件夹名称是小于或等于3个字母的,用*号的话,我们 无法控制文件名的长度,所以我们使用?来进行匹配

for /D %a in(???) do @echo %a

这个@的意思我在【用法讲解1.不带参数】中讲解过,请各位不会的大大,可以回顾看下,谢谢!

C:\>for /D %a in (???) do @echo %a
Ps

C:\>
  1. /R匹配文件
FOR /R [[drive:]path] %variable IN (set) DO command [command-parameters]

    检查以 [drive:]path 为根的目录树,指向每个目录中的 FOR 语句。
    如果在 /R 后没有指定目录规范,则使用当前目录。如果集仅为一个单点(.)字符,
    则枚举该目录树。

for /R 和for /D 的区别是 /D是匹配的文件夹,/R匹配的是文件

for /R 要指定盘符:文件夹  %变量 in (匹配的正则) do 执行的语句

比如说我想匹配c盘下文件夹1下文件名后缀位.txt的文件名,我可以这么写:

for /R c:\1  %a in (*.txt) do @echo %a

效果如下:

C:\>for /R c:\1  %a in (*.txt) do @echo %a
c:\1\123.txt
c:\1\abc.txt

C:\>
  1. /L数字序列
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)

这个的官方的解释比较好理解,我就不再将讲解了,这里我就举个例子吧!
,比如说我想使用【mkdir】创建10个文件夹,文件夹名称从1到10排布。

其中的mkdir是用于创建文件夹的,用法如下

mkdir 创建的文件夹名

我们来用for /L 创建10个文件夹

for /L %a in (1,1,10) do @mkdir %a

我们来使用前面学习的for /D 来验证是否成功创建文件夹,

C:\1>for /D %a in (*) do @echo %a
1
10
2
3
4
5
6
7
8
9

C:\1>

可以看到我们成功创建了文件夹名称位1到10的文件夹。

  1. /F处理文本
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 ["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]

    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 /F 文件操作选项 %变量名 in (文件名) do 执行的命令

这个文件操作选项如下:

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

我先讲解常用的几个参数【skip,delims,tokens】

skip=n      
# 就是选择从文件的第几行开始处理
delims=xxx   
#指定分割符 ,for /F 时将文件每行的内容分割成若干字符串进行处理的。
tokens=x,y,m-n   
#指定处理每行的第几个字符串

这次我将一个综合型的例子吧,正好将前面学的for /L 再复习一次
我们做一个ping扫描程序,将ping成功的ip存入一个名为ip的文本文件中。

@echo off
for /L %%a in (1,1,254)  do ping.exe -n 1 -w 500 192.168.1.%%a >>ping.txt
findstr  "字节=" ping.txt >>ip_list.txt
for /F "tokens=2 delims= " %%b in (ip_list.txt) do echo %%b >>ip.txt
del ping.txt
del ip_list.txt

这里我们先看下百度百科上的解释。
https://baike.baidu.com/item/批处理/1448600?fr=aladdin

批处理(Batch),也称为批处理脚本。顾名思义,批处理就是对某对象进行批量的处理,通常被
认为是一种简化的脚本语言,它应用于DOS和Windows系统中。批处理文件的扩展名为bat 。
目前比较常见的批处理包含两类:DOS批处理和PS批处理。PS批处理是基于强大的图片编辑
软件Photoshop的,用来批量处理图片的脚本;而DOS批处理则是基于DOS命令的,用来自动
地批量地执行DOS命令以实现特定操作的脚本。更复杂的情况,需要使用if、for、goto等命令
控制程式的运行过程,如同C、Basic等高级语言一样。如果需要实现更复杂的应用,利用外部
程式是必要的,这包括系统本身提供的外部命令和第三方提供的工具或者软件。批处理程序虽
然是在命令行环境中运行,但不仅仅能使用命令行软件,任何当前系统下可运行的程序都可以
放在批处理文件中运行。

这个百度百科上的解释有点长,总结来说就是将若干条dos命令写在一个后缀名为。bat的文件中。批处理和dos的最大区别是;批处理定义变量需要比dos多一个%号。
好的,我们回到代码中讲解下每条代码的作用

@echo off
#这行命令的作用是关闭回显,也就是不需要再每次 do语句执行的命令前面加 @ 号了。

for /L %%a in (1,1,254)  do ping.exe -n 1 -w 500 192.168.1.%%a >>ping.txt
#这里我利用for /L 生成一个数字序列,从1开始递增到255,步长为1,并将其依次赋值给变量%%a。
#执行的语句是:是用Ping 命令 ping一个包,并且设置超时时间为500毫秒
#ping的目的地址为:192.168.1.%%a ,也就是说第一次%%a的值为1时,我ping
#的地址就是192.168.1.1,为2时,我ping的就是192.168.1.2,以此类推。
#使用>>将输入保存到一个名为ping.txt的文本文件中。

findstr "字节=" ping.txt >>ip_list.txt
#使用findstr 命令再ping.txt这个文本文件中搜索 字符串"字节="所在的行,并用“>>” 将其保存到文本文件ip_list.txt中

我们可以先看下这个生成的ip_list.txt文件的内容

来自 192.168.1.6 的回复: 字节=32 时间=2ms TTL=64
来自 192.168.1.7 的回复: 字节=32 时间=2ms TTL=64
来自 192.168.1.8 的回复: 字节=32 时间=77ms TTL=64
来自 192.168.1.11 的回复: 字节=32 时间=25ms TTL=64
来自 192.168.1.13 的回复: 字节=32 时间=1ms TTL=64
来自 192.168.1.19 的回复: 字节=32 时间=41ms TTL=64
来自 192.168.1.25 的回复: 字节=32 时间=94ms TTL=255
来自 192.168.1.43 的回复: 字节=32 时间=3ms TTL=128
来自 192.168.1.48 的回复: 字节=32 时间=92ms TTL=128
来自 192.168.1.50 的回复: 字节=32 时间<1ms TTL=255
来自 192.168.1.63 的回复: 字节=32 时间=1ms TTL=64
来自 192.168.1.64 的回复: 字节=32 时间=218ms TTL=64
来自 192.168.1.69 的回复: 字节=32 时间=4ms TTL=64
来自 192.168.1.70 的回复: 字节=32 时间=162ms TTL=64
来自 192.168.1.110 的回复: 字节=32 时间=3ms TTL=64
来自 192.168.1.129 的回复: 字节=32 时间=15ms TTL=128
来自 192.168.1.132 的回复: 字节=32 时间<1ms TTL=128
来自 192.168.1.137 的回复: 字节=32 时间=76ms TTL=64
来自 192.168.1.154 的回复: 字节=32 时间=1ms TTL=128
来自 192.168.1.165 的回复: 字节=32 时间<1ms TTL=64
来自 192.168.1.185 的回复: 字节=32 时间=1ms TTL=64
来自 192.168.1.200 的回复: 字节=32 时间=1ms TTL=64
来自 192.168.1.222 的回复: 字节=32 时间=2ms TTL=128
来自 192.168.1.253 的回复: 字节=32 时间<1ms TTL=255
来自 192.168.1.254 的回复: 字节=32 时间<1ms TTL=255

可以看到我们需要的Ip地址位于每行中以空格作为分割符的第二个字符串中。
好的我们使用for /F 将其提取出来。

for /F "tokens=2 delims= " %%b in (ip_list.txt) do echo %%b >>ip.txt
#使用for /F 选项参数为tokens=2,也就是第二组字符串,delims=空格 ,就是使用空格作为分割符。
#将提取出来的字符串赋值给变量%%b, 
#in (ip_list.txt) 设置操作的文件为ip_list.txt.
#echo %%b >> ip.txt  显示变量%%b的内容并将其存入文件ip.txt中

del ping.txt
del ip_list.txt
#使用del命令分别删除前面生成的临时文件 ping.txt 和 ip_list.txt

写在结尾

博主在最后想说一句:
学习编程也好,学习linux也好,都只是学习使用工具,剩下的就是你如何使用工具来享受这个世界,当你真正意识到这一点时,你离成功已不遥远。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值