突然迷上了windows的批处理,突然习惯了在cmd下启动程序,突然习惯了mkdir,cp,rm.....
禁不住诱惑,用了几天时间仔细的研究了一下bat的语法结构,主要参照了网上的两篇word文档,感觉写的不错:
windows_bat命令编写大全.docx(33kB) windows_BAT批处理命令.doc(175KB)
第一篇word文档主要介绍批处理中一些很常用的命令的用法,每个命令都很细致,学完批处理的简单语法结构之后使用中再看效果会好很多。
第二篇word文档就讲的有点天马行空了。前1/3部分讲的东西简单易懂,比较适合入门;中间1/3讲的比较细致深入,主要有set,choice,if,for,内置符号的用法,是我关注的重点,进阶语法;;最后1/3有一个很庞大的批处理程序,用来优化xp系统的,基本都是注册表的操作,对于学习批处理的价值不大,可略过。
可以先拿第二篇的前1/3练手,了解基本语法结构,然后可以关注中间1/3部分,强化if,for,choice,set等的使用,这些我感觉真的是批处理里面的精华部分了,如果在写自己的批处理的时候遇到不知道怎么办的时候,可以看看第一篇word文档,浏览基本命令,大致知道各个命令能做什么。熟能生巧。
感觉主要是需要知道可以用什么命令去解决,至于命令的语法不知道,直接查看命令帮助就行,帮助是最全的文档。
set:: 定义变量,切割字符串,算术运算
for : 字符串遍历,文件遍历,循环次数控制,切割串
if : 条件判断
goto: 标签跳转
call: bat调用或者label调用
choice: 交互式程序必备
一下内容为个人稍作的总结,以为自己的备忘,个人比较喜欢ue,就直接txt了:
windows bat:
1,echo 表示显示此命令后的字符
2,echo off 表示在此语句后所有运行的命令都不显示命令行本身
3,@与echo off相象,但它是加在每个命令行的最前面,表示运行时不显示这一行的命令行(只能影响当前行)。
4,call 调用另一个批处理文件(如果不用call而直接调用别的批处理文件,那么执行完那个批处理文件后将无法返回当前文件并执行当前文件的后续命令)
5,pause 运行此句会暂停批处理的执行并在屏幕上显示“Press any key to continue...”或“请按任意键继续。。。”的提示,等待用户按任意键后继续。
6,rem 表示此命令后的字符为解释行(注释),不执行,只是给自己今后参考用的(相当于程序中的注释)。
Example:
@echo off
rem echo off usage
pause
7,%[1-9]表示参数,参数是指在运行批处理文件时在文件名后加的以空格(或者Tab)分隔的字符串。变量可以从%0到%9,%0表示批处理命令本身,其它参数字符串用%1到%9顺序表示。
8,::choice: CHOICE [/C choices] [/N] [/CS] [/T timeout /D choice] [/M text]
/M提示文本,/C选项,/T超时事件,/D超时默认选择
CHOICE /C YNC /M "确认请按 Y,否请按 N,或者取消请按 C"
choice /C dme /M "defrag,mem,end"
if errorlevel 3 goto end
if errorlevel 2 goto mem
if errorlevel 1 goto defrag
9,微软里面内置了下列字符不能够在创建的文件名中间使用
con nul aux \ / | || && ^ > < *
特殊符号,要不""括着,要不加^转义
> 创建一个文件
>> 追加到一个文件后面
, 和空格一样的缺省分隔符号.
; 注释,表示后面为注释
: 标号作用
| 管道操作
用这种方法可以同时执行多条命令,而不管命令是否执行成功
dir c:\*.exe & dir d:\*.exe & dir e:\*.exe
&& Usage:第一条命令 && 第二条命令 [&& 第三条命令...]
当碰到执行出错的命令后将不执行后面的命令,如果一直没有出错则一直执行完所有命令;
|| Usage:第一条命令 || 第二条命令 [|| 第三条命令...]
当碰到执行正确的命令后将不执行后面的命令,如果没有出现正确的命令则一直执行完所有命令;
10,if语句即可以实现选择功能。
1、字符串比较
if语句仅能够对两个字符(串)是否相同、先后顺序进行判断等。其命令格式为:
IF [not] string1 compare-op string2 command1 [else command2]
其中,比较操作符compare-op有以下几类:
== - 等于
EQU - 等于
NEQ - 不等于
LSS - 小于
LEQ - 小于或等于
GTR - 大于
GEQ - 大于或等于
选择开关/i则不区分字符串大小写;选择not项,则对判断结果进行逻辑非。
字符串比较示例:
===============================================
@echo off
set str1=abcd1233
set str2=ABCD1234
if %str1%==%str2% (echo 字符串相同!) else (echo 字符串不相同!)
if /i %str1% LSS %str2% (echo str1^<str2) else (echo str1^>=str2)
echo.
set /p choice=是否显示当前时间?(y/n)
if /i not %choice% EQU n echo 当前时间是:%date% %time%
pause>nul
===============================================
对于最后一个if判断,当我们输入n或N时的效果是一样的,都不会显示时间。如果我们取消开关/i,则输入N时,依旧会显示时间。
另外请注意一下几个细节:1-echo str1^<str2和echo str1^>=str2;2-echo.。
2、存在判断
存在判断的功能是判断文件或文件夹是否存在。其命令格式为:
IF [NOT] EXIST filename command1 [else command2]
===============================================
@echo off
if exist %0 echo 文件%0是存在的!
if not exist %~df0 (
echo 文件夹%~df0不存在!
) else echo 文件夹%~df0存在!
pause>nul
===============================================
这里注意几个地方:
1-存在判断既可以判断文件也可以判断文件夹;
2-%0即代表该批处理的全称(包括驱动器盘符、路径、文件名和扩展类型);
3-%~df0是对%0的修正,只保留了其驱动器盘符和路径,详情请参考for /?,属高级批处理范畴;
4-注意if语句的多行书写,多行书写要求command1的左括号必须和if在同一行、else必须和command1的右括号同行、command2的左括号必须与else同行、command1和command2都可以有任意多行,即command可以是命令集。
3、定义判断
定义判断的功能是判断变量是否存在,即是否已被定义。其命令格式为:
IF [not] DEFINED variable command1 [else command2]
存在判断举例:
===============================================
@echo off
set var=111
if defined var (echo var=%var%) else echo var尚未定义!
set var=
if defined var (echo var=%var%) else echo var尚未定义!
pause>nul
===============================================
对比可知,"set var="可以取消变量,收回变量所占据的内存空间。
11,for
1、无开关
无开关的for语句能够对设定的范围内进行循环,是最基本的for循环语句。其命令格式为:
FOR %%variable IN (set) DO command
其中,%%variable是批处理程序里面的书写格式,在DOS中书写为%variable,即只有一个百分号(%);set就是需要我们设定的循环范围,类似于C语言里面的循环变量;do后面的command就是循环所执行的命令,即循环体。
无开关for语句举例:
===============================================
@echo off
for %%i in (a,"b c",d) do echo %%i
pause>nul
===============================================
2、开关/L
含开关/L的for语句,可以根据set里面的设置进行循环,从而实现对循环次数的直接控制。其命令格式为:
FOR /L %%variable IN (start,step,end) DO command
其中,start为开始计数的初始值,step为每次递增的值,end为结束值。当end小于start时,step需要设置为负数。
含开关/L的for语句举例(创建5个文件夹):
===============================================
@echo off
for /l %%i in (1,2,10) do md %%i
pause
===============================================
上例将新建5个文件夹,文件夹名称依次为1、3、5、7、9。可以发现,%%i的结束值并非end的值10,而是不大于end的一个数。
3、开关/F
含开关/F的for语句具有最强大的功能,它能够对字符串进行操作,也能够对命令的返回值进行操作,还可以访问硬盘上的ASCII码文件,比如txt文档等。其命令格式为:
FOR /F ["options"] %%variable IN (set) DO command
其中,set为("string"、'command'、file-set)中的一个;options是(eol=c、skip=n、delims=xxx、tokens=x,y,m-n、usebackq)中的一个或多个的组合。各选项的意义参见for /f。一般情况下,使用较多的是skip、tokens、delims三个选项。
含开关/F的for语句举例:
===============================================
@echo off
echo **No Options:
for /f %%a in ("1,2,10") do echo a=%%a
echo **Options tokens ^& delims:
for /f "tokens=1-3 delims=," %%a in ("1,2,10") do echo a=%%a b=%%b c=%%c
pause
===============================================
@echo off
echo 本文件夹里面的文件有:
for /f "skip=5 tokens=3* delims= " %%a in ('dir') do (
if not "%%a"=="<DIR>" if not "%%b"=="字节" if not "%%b"=="可用字节" echo %%b
)
pause
===============================================
@echo off
echo 本文件夹里面的文件有:
dir>c:\file.txt
for /f "skip=5 tokens=3* delims= " %%a in (c:\file.txt) do (
if not "%%a"=="<DIR>" if not "%%b"=="字节" if not "%%b"=="可用字节" echo %%b
)
del c:\file.txt
pause
===============================================
对于后面的两个例子,其中options里面的delims= 是可以删除的,因为只要添加了/F开关系统就将delims的值默认为空格。
符号字符串中的最后一个字符星号,
那么额外的变量将在最后一个符号解析之后
分配并接受行的保留文本。本例中也可以改为4,不过文件名中有空格的文件,只能显示空格以前部分
同时我们也看到了,for语句的do后面的command也是可以分行的,只需要保证command的左括号和do在同一行就可以了。
4、开关/D或/R
含开关/D或/R的for语句是与目录或文件有关的命令,一般情况下很少使用。含开关/R的命令有时候被用于通过遍历文件夹来查找某一个文件或文件夹,故而列举此例。
含开关/R的for语句举例(文件夹遍历):
===============================================
@echo off
setlocal enabledelayedexpansion
FOR /R d: %%i IN (.) DO (
set dd=%%i
set "dd=!dd:~0,-1!"
echo !dd!
)
pause
exit
===============================================
-----------------------------------2013/5/28 13:11:04-----------------------------------
windowsbat学习总结:
批处理命令返回0为成功, 有些时候看起来命令是正确的,但是始终报错,给命令加上括号尝试一下。
针对if语句,else语句始终加入括号。
1,@, rem, ::, set, echo, pasue, goto, exit, call, choice, if, for,
2,&连续执行,不管正确错误, &&连续执行,直到错误为止, ||连续执行,直到正确为止, >创建文件, >>追加到文件
&, &&, ||,
true & echo good 返回0,errorlevel
false & echo good 返回1,
>, >>
echo good > file ;创建
echo good >> file ;追加
3,choice 语句结构
choice /c ync /t 5 /d y /m "please choice yes, no, cancel"
4,if语句结构
1、字符串比较
set var1="1234A"
set var2="1234a"
if /I %var1% EQU %var2% (echo equal) else (echo not equal)
2、存在判断
if exist %0 (echo exist %0) else (echo not exist %0)
3、定义判断
set var="var"
if defined var echo (var=%var%) else echo "var not defined"
set var=
if defined var echo (var=%var%) else echo "var not defined"
4、choice判断
errorlevel的if判断是大于等于的,请注意,最好的是goto语句跳出去标号比较好
CHOICE /C YNC /M "确认请按 Y,否请按 N,或者取消请按 C"
choice /C dme /M "defrag,mem,end"
if errorlevel 3 echo end
if errorlevel 2 echo mem
if errorlevel 1 echo defrag
5,for语句结构
1、无开关
直接遍历字符串,以空格分隔,当遍历的字符串有空格,会出现"" 也被计入字符串的情况
set list=how do "you suck" do ok
for %%i in (%list%) do echo i=%%i
直接遍历字符串,以,分隔,可以处理字符串自身带有空格的情况
set list=how,do,you suck,do,ok
for %%i in (%list%) do echo i=%%i
2、开关/L
::start为开始计数的初始值,step为每次递增的值,end为结束值。当end小于start时,step需要设置为负数。
::固定循环次数的for循环
for /L %%i in (1, 3, 15) do echo i=%%i
3、开关/F
可以用来截取字符串的某一个部分,以delims字符分隔
for /f %%a in ("1,2,10") do echo a=%%a
for /F "tokens=1-3 delims=," %%a in ("1,2,10") do echo a=%%a b=%%b c=%%c
我感觉自己可能比较常用的for循环:
set list=how do "you suck" do ok ;空格分隔的字符串拆分
for %%i in (%list%) do echo i=%%i
set list=how,do,you suck,do,ok ;逗号分隔的字符串拆分,去掉了引号的问题
for %%i in (%list%) do echo i=%%i
set list=how do you suck do ok
for %%i in (%list%) do echo i=%%i
for /L %%i in (1, 3, 15) do echo i=%%i ;控制次数的for循环
for /F "tokens=1-3 delims=," %%a in ("1,2,10") do echo a=%%a b=%%b c=%%c ;截取串的for语句,不应该被看作是for循环
cls
高级用法:
skip表明跳过前5行,tokens表明选取第三列作为%%i,后面的全部作为%%j,if判断可以通过语句的方式实现多个条件连接判断。
@echo off
echo 本文件夹文件有:
for /F "skip=5 tokens=3* delims= " %%i in ('dir') do (
::echo "i=%%i, j=%%j"
if not "%%i"=="<DIR>" if not "%%j"=="字节" if not "%%j"=="可用字节" (
echo %%j
)
)
-----------------------------------2013/5/28 13:46:42-------------------------------
使用情况:
bat:
set var=%str::=2% 则表示把字符串abc:de中的:替换为2;
获取字符串长度
set str=teststring
:nextlost
if not "%str%"=="" (
set /a num+=1
set "str=%str:~1%"
goto nextlost
)
Test.bat: 传入的参数为参数:c:\temp\test1.txt
认为后面的1应该指的是第一个参数
echo %~d1 =>c:
echo %~dp1 =>c:\temp\
echo %~nx1 =>test1.txt
echo %~n1 =>test1
echo %~x1 =>.txt
cho 当前目录路径:%~dp0
语音:
mshta vbscript:createobject("sapi.spvoice").speak("好好学习Merry Christmas and Happy New Year!")(window.close)
for循环遍历用空格间隔的字符串,对每一个字符串做相应的处理。
:nextfor
FOR /F "tokens=1,*" %%i in ("%serviceslist%") do (
set torestart=%%i
set serviceslist=%%j
if not "%serviceslist%"=="" (
call :restartprog
goto :nextfor
)
)
goto :end
::函数调用演示
::-----------restartprog begin---------------
:restartprog
echo "restartprog: %torestart%"
goto :EOF
::-----------restartprog end----------------
set计算求值:
set var=10
set /a var+8*2+var*2
set /a "var+8*2+var*2>>2"
延迟环境变量扩展:
setlocal enabledelayedexpansion
set VAR=before
if "%VAR%" == "before" (
set VAR=after
if "!VAR!" == "after" @echo If you see this, it worked
)
命令扩展???:还不清楚
echo %random%
echo %cd%
set v
set p
截取字符串:
set var=10203040
第一数字是位置,第二个位置是长度,第二个位置如果无,默认是结尾,第一个位置如果是负数,则反方向的位置。位置从0开始
echo %var:~-4,3%
echo %var:~0%
echo %var:~1%
echo %var:~-2%
替换字符串
echo %var:0=kkk% ;0替换为kkk
echo %var:10=kkk% ;10替换为kkk
echo %var:20=kkk%
echo %var:*20=kkk% ;20之前包括20字符串都替换为kkk