zoukankan      html  css  js  c++  java
  • 探索Windows命令行系列(3):命令行脚本基础

    命令行脚本是包含可执行命令的文本文件,这些命令一般和命令行窗口键入的命令一致。脚本文件中的命令就是标准的文本字符,因此可以通过普通的文本编辑器,如 Windows 记事本来编辑。在脚本中输入命令时,需要注意的是,每一条命令,或者每一组需要一起执行的命令,都因该保存在脚本的单独一行中,以便确保命令的正确执行。

    在常见的编程语言中,如 C/C++/C# 等语言中每条语句都要用分号来表示结束。而在命令行脚本中,没有诸如分号之类的特定终止符,而是通过断行(比如按下Shift+Enter键)、回车与断行(按下Enter键)、文件结尾标志三种方式来表示语句结束。

    1、实用功能

    1.1、为脚本添加注释

    rem 命令用于注释单行命令,类似与 C# 等语言中注释符//。需要注意的是,当在批处理文件中通过 rem 命令注释了某一行之后,该行就不会被执行了,但默认被注释的命令内容还是会被回显到窗口上,这时候可以结合下面将要讲到的 @ 命令,写成 @rem 就一定不会显示了。

    1.2、控制命令的回显

    echo@两个命令,echo命令用于在命令回显消息、关闭或打开命令回显(当回显处于打开状态,执行批处理文件时将逐条显示批处理文件中的命令行;相反,当回显处于关闭状态时,不会显示命令行本身),而@命令用于以行为单位控制命令的回显方式。

    示例 1:向命令行回显一条消息(Hello Word!)

    echo Hello Word!
    

    示例 2:关闭命令行回显(关闭回显之后就看不到命令提示符前的路径了)

    echo off
    

    示例 3:打开命令行回显(默认打开)

    echo on
    

    示例 4:在任何一行命令前加上@即可关闭该行的回显

    @echo Hello Word! 
    

    1.3、使用数学表达式

    命令行脚本支持算术运算、赋值运算、比较运算、按位运算等常见的数学运算。所有数学运算都得在set命令的参数/a之后进行。

    示例:

    set /a add=3+7
    set /a sub=3-7
    set /a mul=3*7
    set /a div=5/2
    set /a mod=5%2
    set /a exp=2*2*2
    

    1.4、向脚本传递参数

    在调用命令行脚本的时候还可以向脚本传递参数。传递给脚本的每一个参数都可以用形参进行检查和解释。第1个参数用%1表示,第2个参数用%2表示,依次类推,第9个形参为%9。但如果向脚本传递的参数多于9个,则会被保存到一个特殊的参数变量%*中去,通过shift命令可以查看%*中的实参;另外在必要的时候,还可以通过shift命令进行形参移位,譬如shift /2表示忽略第2个参数,从第3个参数开始往前移动。还有一个特殊的形参是%0,它代表当前脚本的名称。

    示例:

    在 D 盘新建一个记事本,然后键入下面的脚本:

    @echo 当前脚本的名称是:%0
    @echo 第1个参数的值是:%1
    @echo 第2个参数的值是:%2
    @pause
    

    将记事本改名为 try.bat,然后打开命令行(cmd.exe)调用该脚本:

    try abc 123
    

    结果如下:
    cmd03

    2、使用变量

    在命令行脚本中,我们通常所说的变量更可能是指环境变量。环境变量有多种来源,有内置在操作系统之中的,也有在系统启动时来自系统硬件的,这些被称之为内置系统变量,对所有 Windows 进程都可用。系统变量也可来自 Windows 注册表。除内置系统变量之外,还有一些登录时设置的变量,这种变量被称为内置用户变量。

    通过set命令可以查看当前命令行实例中所有变量及变量值。除标准的系统变量和用户变量之外,还可以通过set命令在命令行中随时创建或设置变量。

    有些变量在命令行中有着特殊的含义,如 path、computername、homedrive、errorlevel 等。其中 errorlevel 变量可以追踪最近使用命令的退出代码。如果命令正常执行,则错误级别为 0;如果命令在执行过程中出错,则错误级别会被设置为一个适当的非 0 值(1 表示一般的错误,2 表示执行错误,意味着命令没有正确执行,-2 表示算数错误)。

    2.1、变量的命名及定义

    在命令行脚本中,变量名是大小写不敏感的,在定义变量的时候可以区分大小写,但在调用的时候不区分大小写。变量名可以是任意字母、数字或字符的组合,如 2s、38、? 等理论上都是合法的变量名。尽管如此,我们也还是应该要使用便于理解的名字来做为变量名。

    在命令行中声明变量时必须同时为其赋值,定义变量的语法如下:

    set variable_name=variable_value
    

    在变量名和变量值中,空格都是有效的,但一般不建议使用。命令行也不对具体的数据类型进行区分,所有的变量都是以字符的形式存储,即使变量值设置为数值也是如此。但在进行数学运算的时候,命令行是可以判断出字符串是否为数值的。

    有些字符是命令行的保留字符,如 @、<、>、&、|、^ 等,在使用这些字符的时候,无论出现在变量值中的那个位置,都必须使用转码字符 ^ 对其进行转码。

    示例:

    set var1=abc
    set var2=985
    set var3=3.14
    set var4="ok"
    set var5="2^&3"
    set "name=张三"
    

    2.2、调用变量

    在命令行中任何需要用到变量的地方都可以调用变量,调用的方式是用两个百分号包裹变量名,即%variable_name%

    示例:

    echo %var3%                       输出变量 var3 的值
    echo %name%                       输出变量 name 的值
    cd %systemroot%system32          切换到系统目录下的 system32 目录中
    set syspath=%systemroot%system32 定义一个变量 syspath 并用其它变量为其赋值
    

    综合案例:

    @rem 定义变量接收用户输入
    @echo off
    @set /p str=请输入任意内容,然后回车:
    @echo 您刚刚输入的内容是:%str%
    @pause
    

    2.3、变量的作用域

    在命令行中用set命令所做修改的作用范围是局部化的,也就是说set命令的作用仅限当前会话窗口,窗口关闭后之前的修改就会失效。

    有时候可能还需要对变量的作用范围做进一步约束,使其仅作用于脚本的某一部分。要做到这一点,可以通过setlocal命令来标记一个局部范围的开始,使用endlocal命令来标记一个局部范围的结束。

    示例:

    setlocal
    set cnt=5
    endlocal
    

    在一个比较长的作用范围内,使用完变量之后,应该要对其进行适当处理,以便释放该变量占用的内存,防止无意间再次引用该变量时出现预期之外的结果。在命令行中只要把变量的值设置为空,该变量就不复存在了。

    示例:

    set var3=
    

    3、结构语句

    3.1、选择语句

    命令行中的选择语句是 if,该语句有 3 中不同的形式,本节将逐一介绍。

    普通的 if 语句:如果条件为真则执行该语句,否则绕过该语句。

    语法(语法中的圆括号并非必须,但使用圆括号会使语句结构更加清晰):

    if condition (statement1) [else (statement2)]
    

    示例:

    set v1=1
    if %v1%==1 (echo is one) else (echo is not one)
    if %v1%==1 (hostname & ver & ipconfig /all)
    

    如果存在 try.bat 文件,则把该文件的内容输出到控制台:

    if exist try.bat type try.bat
    

    if not 语句:如果条件为假则执行该语句,否则绕过该语句。

    语法:

    if not condition (statement1) [else (statement2)]
    

    示例:

    if not %errorlevel%==0 (echo error has occurred) else (echo running well)
    

    if defined & if not defined:这两个语句用于检查某变量是否存在。

    语法:

    if defined variable_name statement
    if not defined variable_name statement
    

    在用于安全检查的时候,这两个语句非常有用。第一个语句是在变量存在的时候,执行某条命令。第二个语句是在变量不存在的时候,执行某条命令。

    示例:

    if defined v1 (echo %v1%)
    

    3.2、迭代语句

    命令行中的迭代语句是 for,该语句与常见的编程语言(如 C#)中的 for 循环有很大的不同。命令行中的 for 语句主要用于在成组的文件与目录中进行迭代处理,并以行为基础分析文本文件、字符串以及命令的输出信息。

    最常见的 for 语句语法如下:

    for iterator do (statement)
    

    语法中的 iterator 用于控制 for 循环的执行。对 iterator 中每一个步骤或元素,都会执行特定的语句。它可以是单一的一条命令,也可以是使用命令管道、命令链、命令分组等技术组合起来的多条命令。

    iterator 通常包含一个初始化变量和一组需要反复执行的元素,比如需要遍历的一组文件或某范围内的一组值。初始化变量也就是要使用的值的占位符,使用初始化变量时,需要注意以下几点:

    • 变量只存在于 for 语句的上下文中。
    • 变量名必须在 a~z 或 A~Z 的范围内,比如 %%a、%%b、%%c。
    • 变量名是大小写敏感的,也就是说,%%a 与 %%A 是不同的。

    常见迭代形式表如下:

    迭代功能 语法格式
    文件集合 for %%variable in (fileSet) do statement
    目录集合 for /D %%variable in (directorySet) do statement
    子目录中的文件 for /R[path] %%variable in (fileSet) do statement
    遍历一系列的值 for /L %%variable in (stepRange) do statement
    分析文本文件、字符串以及命令输出 for /F[options] %%variable in (source) do statement

    上表中提供的迭代形式也可以在命令行中交互式地使用 for 语句,这种情况下,应该使用 %variable,而不是 %%variable。除此之外,脚本中的 for 语句与在命令行中使用 for 语句是完全一样的。

    遍历一系列值:用于操作数值。

    语法:

    for /l %%variable in (start,step,end) do (statement)
    

    示例(输出 1~9 之间的奇数):

    @for /l %%v in (1,2,9) do @echo %%v
    

    在成组的文件中迭代执行:用于操作文件组中的文件。

    语法:

    for %%variable in (fileSet) do statement
    

    语法中的 fileSet 用于指定需要处理的文件集,文件集可以是如下形式。

    • 通过文件名指定为单独的文件,如 myfile.txt。
    • 通过文件名通配符指定一组文件,如 *.txt。
    • 使用空格分割的多个文件或文件组,如 myfile.txt *.doc *.docx。

    示例 1(输出 D:Workspace 目录中所有 txt 文件的全路径名):

    for %%i in (D:Workspace*.txt) do (echo %%i)
    

    示例 2(输出 D:Workspace 目录及其子目录中所有 txt 文件的全路径名):

    for /r D:Workspace %%i in (*.txt) do (echo %%i)
    

    示例 3(输出 D:Workspace 目录及其子目录中所有 jpg 或 png 文件的全路径名):

    for /r D:Workspace %%i in (*jpg,*png) do (echo %%i)
    

    示例 4(输出 D:Workspace 目录及其子目录中所有文件的全路径名):

    for /r D:Workspace %%i in (*) do (echo %%i)
    

    示例 5(输出 D:Workspace 目录及其子目录中所有名称包含 windows 的文件全路径名):

    for /r D:Workspace %%i in (*windows*) do (echo %%i)
    

    示例 6(输出 D:Workspace 目录及其子目录中所有路径包含 windows 的文件夹全路径+windows):

    for /r D:Workspace %%i in (windows) do (echo %%i)
    

    从上述几个例子中也能看出来,匹配模式中第一个字符必须是星号,否则就匹配不到文件了。如果完全不带星号则会匹配到路径包含指定字符的文件夹全路径+指定字符。

    在目录中迭代执行:用于操作目录组中的目录。

    示例(输出 D:Workspace 中所有的子目录名):

    for /d %%v in (D:Workspace*) do echo %%v
    

    3.3、子程序 & 过程

    使用子程序:也就是通过goto跳转到另一段程序继续执行。goto的作用是将命令解释器直接跳转到批处理脚本中某个标记行。

    示例:

    @if not defined v (@echo error: Undefined parameters!) & (goto exit)
    @if %v%==1 goto subroutine1
    @if %v%==2 goto subroutine2
    @if %v%==3 goto subroutine3
    @goto exit
    
    :subroutine1
    @echo In subroutine 1
    @goto exit
    
    :subroutine2
    @echo In subroutine 2
    @goto exit
    
    :subroutine3
    @echo In subroutine 3
    @goto exit
    
    :exit
    @echo Exiting...
    

    使用过程:也就是在脚本中调用其它脚本,而又不会退出原脚本。进行过程调用时,命令行会执行调用的脚本,逐一执行其中的命令,结束后返回到原脚本,并从过程调用语句的下一行语句开始执行。

    示例:

    @if %v%==1 call file1
    

    上例中,如果不适用call语句,也能调用 file1 脚本,但在 file1 脚本执行完毕之后控制权不会返回给调用者(原脚本)。

    call 命令用于调用另外一个批处理文件。常用的方式就是call sample.bat,新扩展的一个用法是call :label arguments,也就是用 call 来调用 bat 文件内部的一段脚本,然后用 exit 来退出当前的 call 调用。

    4、总结

    本文主要讲述了命令行脚本的注释、回显、参数、变量、结构控制语句等基础内容。

    本文链接http://www.cnblogs.com/hanzongze/p/cmd-script-base.html
    版权声明:本文为博客园博主 韩宗泽 原创,作者保留署名权!欢迎通过转载、演绎或其它传播方式来使用本文,但必须在明显位置给出作者署名和本文链接!个人博客,能力有限,若有不当之处,敬请批评指正,谢谢!

  • 相关阅读:
    转:WinCE驱动开发问题精华集锦
    转:WINCE6.0+S3C6410下的DM9000A驱动
    转:WinCE6.0 不重启修改IP地址
    前端小功能:canvas验证码
    前端小功能:canvas签名版
    JavaScript中数组的应用方式
    ES6--Promise
    wepy 语法高亮
    Javascript和JQuery获取浏览器窗口各种尺寸
    图片预加载
  • 原文地址:https://www.cnblogs.com/hanzongze/p/cmd-script-base.html
Copyright © 2011-2022 走看看