批处理语句 set var=!var:你好=!

参数%0具有特殊的功能,可以调用批处理自身,以达到批处理本身循环的目的,也可以复制文件自身等等。

例:最简单的复制文件自身的方法 

在bat文件中,多个命令写在一行或括号内,如果使用行内或括号内定义或修改的变量,需要启用延迟变量扩充,用叹号引用变量。

ren命令中的文件名最好加引号,防止文件名包含空格等特殊字符造成的错误。

批处理文件中的一条语句
意思是 更改当前目录为批处理本身的目录 

%0代指批处理文件自身
%~d0 是指批处理所在的盘符
cd %~dp0 就是进入批处理所在目录了

%cd%代表当前执行批处理文件的路径(当前路径)

%CD%  当前文件目录,不显示文件名,切通过其找到其他文件

%0    当前文件本身,包含完整路径名和文件名

>和>>它们的作用都是改变各种提示信息的输出方向,把提示信息输出到指定的地方去——各种提示信息默认是输出到屏幕上去的。

单个的>表示以覆盖方式重定向提示信息,也就是说如果目的地原来有内容,将把原来的内容清除掉,用新内容填充;

>>表示以追加的方式重定向提示信息,也就是说,如果目的地原来有内容,原有的内容将保持不变,在原有内容之后追加新内容。

help>nul 就把 help 命令的提示信息重定向到了空设备中去(nul表示空设备)。

格式:第一条命令 | 第二条命令 [| 第三条命令...]

将第一条命令的结果作为第二条命令的参数来使用,记得在unix中这种方式很常见。

以上命令是:查找C:\所有,并发现TXT字符串。

  如果要创建的文件夹带有空格或&,需要用引号把文件夹名括起来,例如:md "test abc"、md "abc&xyz"。

  如果不使用引号,又会带来什么后果呢?

  1、如果文件夹名带空格,那么,md test abc 语句会在当前目录下创建test和abc这两个文件夹;利用这个特点,如果要创建abc def xyz这三个文件夹,直接使用 md abc def xyz 就行了,而无需连写三条md语句。

  2、如果文件夹名中含有&,那么,md abc&xyz 会创建abc这个文件夹,并提示说:'xyz'不是内部或外部命令,也不是可运行的程序或批处理文件,这是因为,&是复合语句的连接符号,它把前后两部分视为两条子语句了。

  忠告:如果文件夹名含有特殊符号,请不要忘记使用双引号!

md创建中级目录。也就是说,md a\b\c这样的命令,可以在当前目录下建立文件夹a,然后,在a下建立文件夹b,b之下再建立文件夹c

on 和 off 都是echo 的关键字,所谓关键字就是系统规定、有特殊用途的字符。

系统默认是echo on,所以每次为了关闭回显,都会在代码的行首加上echo off,加以关闭。

不关闭回显,连执行的命令符都会显示出来,所以一般会关闭。

前面加上@又是为什么?

echo off只能关闭 echo off后面的代码的回显,但不能关闭自身的回显。所以需要加上@关闭自身。

echo 后加一点表示换行,真奇葩的设计。 

findstr默认是区分大小写的(跟find命令一样)

不会等upload.bat执行完再执行下一条, 如此可实现同时执行多条命令.

call 命令 是要等调用的程序结束以后才运行下面的命令.

用这种方法可以同时执行多条命令,而不管命令是否执行成功。

当碰到执行出错的命令后将不执行后面的命令,如果一直没有出错则一直执行完所有命令。

||:第一条命令 || 第二条命令 [|| 第三条命令...]

当碰到执行正确的命令后将不执行后面的命令,如果没有出现正确的命令则一直执行完所有命令;

执行后在当前路径下就生成for.txt、set.txt和shift.txt三个文件,里面分别记录了for命令、set命令和shift命令的帮助信息

打开回显或关闭请求回显功能,或显示消息。
如果没有任何参数,echo 命令将显示当前回显设置。

指定跳转到标签,找到标签后,程序将处理从下一行开始的命令。
语法:goto label (label是参数,指定所要转向的批处理程序中的行。) 

标签的名字可以随便起,但是最好是有意义的字母啦,字母前加个:用来表示这个字母是标签,goto命令就是根据这个:来寻找下一步跳到到那里。最好有一些说明这样你别人看起来才会理解你的意图啊。

注释可以使用Rem,也可以使用双冒号即::

在这个例子中,驱动器 A 中磁盘上的所有文件均复制到d:\back中。显示的注释提示您将另一张磁盘放入驱动器 A 时,pause 命令会使程序挂起,以便您更换磁盘,然后按任意键继续处理。

call命令用来从一个批处理脚本中调用另一个批处理脚本

如果在脚本或批处理文件外使用 Call,它将不会在命令行起作用。

调用外部程序,所有的DOS命令和命令行程序都可以由start命令来调用。

MIN 开始时窗口最小化
HIGH 在 HIGH 优先级类别开始应用程序
WAIT 启动应用程序并等候它结束
parameters 这些为传送到命令/程序的参数
执行的应用程序是 32-位 GUI 应用程序时,CMD.EXE 不等应用程序终止就返回命令提示。如果在命令脚本内执行,该新行为则不会发生。

choice 此命令可以让用户输入一个字符,从而运行不同的命令。使用时应该加/c:参数,c:后应写提示可输入的字符,之间无空格。它的返回码为1234……

此文件运行后,将显示 defrag,mem,end[D,M,E]? 用户可选择d m e ,然后if语句将作出判断,d表示执行标号为defrag的程序段,m表示执行标号为mem的程序段,e表示执行标号为end的程序段,每个程序段最后都以goto end将程序跳到end标号处,然后程序将显示good bye,文件结束。

if 表示将判断是否符合规定的条件,从而决定执行不同的命令。 有三种格式: 

检测数值:(注意,批处理中大于符号不能用:“>”,而用"gtr",其它的也类似) 

比较运算符一览: 

for 命令是一个比较复杂的命令,主要用于参数在指定的范围内循环执行命令。
在批处理文件中使用 FOR 命令时,指定变量请使用 %%variable

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

//这个会显示a.txt里面的内容,因为/f的作用,会读出a.txt中的内容。

//而这个只会显示a.txt这个名字,并不会读取其中的内容。

::显示C盘根目录下所有非隐藏、非系统属性文件

::只显示文件,不显示文件夹

(只搜索文件,不搜索目录)

(搜索指定路径及所有子目录中与set相符合的所有文件)   

--枚举d/backup目录,现在当前路径和当前路径下所有文件夹,不过后面带了个1   

  该集表示以增量形式从开始到结束的一个数字序列。可以使用负的 Step   

解说:为什么是4而不是5呢?在echo之前明明已经把变量a的值改成5了?

::这种情况a直接是5

让我们先了解一下批处理运行命令的机制:

批处理读取命令时是按行读取的(另外例如for命令等,其后用一对圆括号闭合的所有语句也当作一行),在处理之前要完成必要的预处理工作,这其中就包括对该行命令中的变量赋值。我们现在分析一下例1,批处理

在运行到这句“set a=5 & echo %a%”之前,先把这一句整句读取并做了预处理——对变量a赋了值,那么%a%当然就是4了!(没有为什么,批处理就是这样做的。)

而为了能够感知环境变量的动态变化,批处理设计了变量延迟。简单来说,在读取了一条完整的语句之后,不立即对该行的变量赋值,而会在某个单条语句执行之前再进行赋值,也就是说“延迟”了对变量的赋值。

那么如何开启变量延迟呢?变量延迟又需要注意什么呢?举个例子说明一下:

解说:启动了变量延迟,得到了正确答案。变量延迟的启动语句是“setlocal enabledelayedexpansion”,并且变量要用一对叹号“!!”括起来(注意要用英文的叹号),否则就没有变量延迟的效果。

分析一下例2,首先“setlocal enabledelayedexpansion”开启变量延迟,然后“set a=4”先给变量a赋值为4,“set a=5 & echo !a!”这句是给变量a赋值为5并输出(由于启动了变量延迟,所以批处理能够感知到动态变化,即不是先给该行变量赋值,而是在运行过程中给变量赋值,因此此时a的值就是5了)。

再举一个例子巩固一下。

(若不加延迟变量,则会输出几个!a!字符)

解说:本例开启了变量延迟并用“!!”将变量扩起来,因此得到我们预期的结果。如果不用变量延迟会出现什

么结果呢?结果是这样的:

ECHO 处于关闭状态。

ECHO 处于关闭状态。

ECHO 处于关闭状态。

ECHO 处于关闭状态。

ECHO 处于关闭状态。

即没有感知到for语句中的动态变化。


字符串查找增强:findstr

设置文件属性:attrib

显示磁盘卷标和序列号:vol

编辑磁盘卷标:label

转换磁盘系统格式:convert

创建虚拟盘符:subst

给磁盘做个体检:chkdsk

清理CMD屏幕上的信息:cls

获取系统日期:date

获取系统时间:time

打开文件、运行程序:start

关闭或重启计算机:shutdown

配置计算机网络信息:netsh

调用子过程或外部程序:call

显示提示信息:echo

注释语句:rem和::

更改参数的位置:shift

一次只显示单屏内容:more

}

当我第一眼看到这个问题的时候, 其实我是拒绝的, 甚至想点举报.

后来想想这是自己为数不多的能回答的问题(好悲伤), 还是算了.

为什么呢, 第二次输出的时候明明已经更改了n的值啊

究其原因是批处理有一个"预处理"机制:

在读入一段代码(被括号包裹/用&连接)时,
cmd会现将形如%n%的变量扩展(意为替换成具体值)

但有时候会带来麻烦, 尤其是对于新手, 尤其是在for中.

上面是一个用很蠢的方法求1~100中的偶数的和的程序, 然而你实际运行这段代码, 就会发现会输出sum=0

???不可思议, 其实这几乎是每个新手必遇到的bug

冷静分析一下, cmd首先读入整个语句, 然后进行预处理操作, 由于r的值未定义, r会被扩展为空值, 也就是变成下面这个样子

显然这个if无论如何都不会成立

解决这个问题的方法很简单, 开启变量延迟, 同时将需要延迟扩展的变量用!包裹

这样r就会在它真正被用到的时候扩展

更详尽的资料见(强烈安利bathome, 这绝对是国内批处理水平最高的社区):

在论坛发帖前请仔细阅读版规

在论坛发帖前请仔细阅读版规

在论坛发帖前请仔细阅读版规

这个属于高阶用法, 一般只会在脚本有一定规模的时候用到, 只是日常写个几十行完成小任务就不必看下去了

setlocal的另一大作用是配合call实现批处理中的"函数"

批处理是没有变量作用域这种设定的, 不管在哪里定义的变量都是全局的, 这使得在批处理中实现函数非常麻烦, 因为你不知道你啥时候就覆盖了外部变量...

setlocal可以开启一个新环境/变量空间, 这个环境继承上个环境的变量, 但是这个环境随时都可以由endlocal销毁, 从而保证了函数不会影响外部变量

常见用法如下, 当然下面这个"函数"非常简单, 实际操作中可能比这个复杂

多棒的功能! 只是可惜setlocal有最大递归层数为32的限制

(不过大部分情况下遇到这个错误的时候都是你代码出了问题...)

PS. 答主目前主力系统是Linux, 已经很久没写批处理了, 又懒得开虚拟机

所以上面的代码包括结果是有可能会出错的(逃

}

  第 4 行:~1,2 表示:把源变量src的值字符串从第1个索引位开始,取2位组成目标变量des的值(即:ww)。

  第 6 行:~4,5 表示:把源变量src的值字符串从第4个索引位开始,取5位组成目标变量des的值(即:baidu)。

  第 8 行:同理第4、6行(结果即:.cn)。

  第 12 行:~-5 表示:把源变量src的值字符串从尾部开始取5个字符组成目标变量des的值(即:)。  

}

我要回帖

更多关于 批处理goto命令 的文章

更多推荐

版权声明:文章内容来源于网络,版权归原作者所有,如有侵权请点击这里与我们联系,我们将及时删除。

点击添加站长微信