DOS批处理高级教程 第七章 DOS批处理编程高级技巧
一、交互界面设计
没啥说的,看看高手设计的菜单界面吧:
@echooff cls title终极多功能修复 :menu cls color0A echo. echo============================== echo请选择要进行的操作,然后按回车 echo============================== echo. echo1.网络修复及上网相关设置,修复IE,自定义屏蔽网站 echo. echo2.病毒专杀工具,端口关闭工具,关闭自动播放 echo. echo3.清除所有多余的自启动项目,修复系统错误 echo. echo4.清理系统垃圾,提高启动速度 echo. echoQ.退出 echo. echo. :cho setchoice= set/pchoice=请选择: IFNOT"%choice%"==""SETchoice=%choice:~0,1% if/i"%choice%"=="1"gotoip if/i"%choice%"=="2"gotosetsave if/i"%choice%"=="3"gotokaiji if/i"%choice%"=="4"gotoclean if/i"%choice%"=="Q"gotoendd echo选择无效,请重新输入 echo. gotocho
只要学完本教程前面的章节,上面的程序应该能看懂了。
二、if…else…条件语句
前面已经谈到,DOS条件语句主要有以下形式
IF[NOT]ERRORLEVELnumbercommand
IF[NOT]string1==string2command
IF[NOT]EXISTfilenamecommand
增强用法:IF[/I]string1compare-opstring2command
增强用法中加上/I就不区分大小写了!
增强用法中还有一些用来判断数字的符号:
EQU-等于
NEQ-不等于
LSS-小于
LEQ-小于或等于
GTR-大于
GEQ-大于或等于
上面的command命令都可以用小括号来使用多条命令的组合,包括else子句,组合命令中可以嵌套使用条件或循环命令。
例如:
IFEXISTfilename(
delfilename
)ELSE(
echofilenamemissing
)
也可写成:
ifexistfilename(delfilename)else(echofilenamemissing)
但这种写法不适合命令太多或嵌套命令的使用。
三、循环语句
1、指定次数循环
FOR/L%variableIN(start,step,end)DOcommand[command-parameters]
组合命令:
FOR/L%variableIN(start,step,end)DO(
Command1
Command2
……
)
2、对某集合执行循环语句。
FOR%%variableIN(set)DOcommand[command-parameters]
%%variable 指定一个单一字母可替换的参数。
(set) 指定一个或一组文件。可以使用通配符。
command 对每个文件执行的命令,可用小括号使用多条命令组合。
FOR/R[[drive:]path]%variableIN(set)DOcommand[command-parameters]
检查以[drive:]path为根的目录树,指向每个目录中的
FOR语句。如果在/R后没有指定目录,则使用当前
目录。如果集仅为一个单点(.)字符,则枚举该目录树。
同前面一样,command可以用括号来组合:
FOR/R[[drive:]path]%variableIN(set)DO(
Command1
Command2
……
commandn
)
3、条件循环
上面的循环结构是用for命令来实现的,for命令循环有一个缺点,就是整个循环被当作一条命令语句,涉及到变量延迟的问题。
利用goto语句和条件判断,dos可以实现条件循环,很简单啦,看例子:
例:
@echooff setvar=0 rem************循环开始了 :continue set/avar+=1 echo第%var%次循环 if%var%lss100gotocontinue rem************循环结束了 echo循环执行完毕 pause
例:
@echooff setvar=100 rem************循环开始了 :continue echo第%var%次循环 set/avar-=1 if%var%gtr0gotocontinue rem************循环结束了 echo循环执行完毕 pause
四、子程序
在批处理程序中可以调用外部可运行程序,比如exe程序,也可调用其他批处理程序,这些也可以看作子程序,但是不够方便,如果被调用的程序很多,就显得不够简明了,很繁琐。
在windowsXP中,批处理可以调用本程序中的一个程序段,相当于子程序,这些子程序一般放在主程序后面。
子程序调用格式:
CALL:labelarguments
子程序语法:
:label
command1
command2
......
commandn
goto:eof
在子程序段中,参数%0指标签:label
子过程一般放在最后,并且注意在主程序最后要加上exit或跳转语句,避免错误的进入子过程。
子程序和主程序中的变量都是全局变量,其作用范围都是整个批处理程序。
传至子程序的参数在call语句中指定,在子程序中用%1、%2至%9的形式调用,而子程序返回主程序的数据只需在调用结束后直接引用就可以了,当然也可以指定返回变量,请看下面的例子。
子程序例1:
@echooff call:subreturn你好 echo子程序返回值:%return% pause :sub set%1=%2 goto:eof
运行结果:你好
子程序例2:设计一个求多个整数相加的子程序
@echooff setsum=0 call:subsum102035 echo数据求和结果:%sum% pause :sub rem参数1为返回变量名称 set/a%1=%1+%2 shift/2 ifnot"%2"==""gotosub goto:eof
运行结果:65
在win98系统中,不支持上面这种标号调用,须将子程序单独保存为一个批处理程序,然后调用。
五、用ftp命令实现自动下载
ftp是常用的下载工具,ftp界面中有40多个常用命令,自己学习了,不介绍了。这里介绍如何用dos命令行调用ftp命令,实现ftp自动登录,并上传下载,并自动退出ftp程序。
其实可以将ftp命令组合保存为一个文本文件,然后用以下命令调用即可。
ftp-n-s:[[drive:]path]filename
上面的filename为ftp命令文件,包括登录IP地址,用户名、密码、操作命令等
例:
open90.52.8.3#打开ip useriware#用户为iware password8848#密码 bin#二进制传输模式 prompt cdtmp1#切换至iware用户下的tmp1目录 pwd lcdd:\download#本地目录 mget*#下载tmp1目录下的所有文件 bye#退出ftp
六、用7-ZIP实现命令行压缩和解压功能
语法格式:(详细情况见7-zip帮助文件,看得头晕可以跳过,用到再学)
7z<command>[<switch>...]<base_archive_name>[<arguments>...]
7z.exe的每个命令都有不同的参数<switch>,请看帮助文件
<base_archive_name>为压缩包名称
<arguments>为文件名称,支持通配符或文件列表
其中,7z是至命令行压缩解压程序7z.exe,<command>是7z.exe包含的命令,列举如下:
a:Addsfilestoarchive.添加至压缩包
a命令可用参数:
-i(Include)
-m(Method)
-p(SetPassword)
-r(Recurse)
-sfx(createSFX)
-si(useStdIn)
-so(useStdOut)
-ssw(Compresssharedfiles)
-t(Typeofarchive)
-u(Update)
-v(Volumes)
-w(WorkingDir)
-x(Exclude)
b:Benchmark
d:Deletesfilesfromarchive.从压缩包中删除文件
d命令可用参数:
-i(Include)
-m(Method)
-p(SetPassword)
-r(Recurse)
-u(Update)
-w(WorkingDir)
-x(Exclude)
e:Extract解压文件至当前目录或指定目录
e命令可用参数:
-ai(Includearchives)
-an(Disableparsingofarchive_name)
-ao(Overwritemode)
-ax(Excludearchives)
-i(Include)
-o(SetOutputDirectory)
-p(SetPassword)
-r(Recurse)
-so(useStdOut)
-x(Exclude)
-y(AssumeYesonallqueries)
l:Listscontentsofarchive.
t:Test
u:Update
x:eXtractwithfullpaths用文件的完整路径解压至当前目录或指定目录
x命令可用参数:
-ai(Includearchives)
-an(Disableparsingofarchive_name)
-ao(Overwritemode)
-ax(Excludearchives)
-i(Include)
-o(SetOutputDirectory)
-p(SetPassword)
-r(Recurse)
-so(useStdOut)
-x(Exclude)
-y(AssumeYesonallqueries)
七、调用VBScript程序
使用Windows脚本宿主,可以在命令提示符下运行脚本。CScript.exe提供了用于设置脚本属性的命令行开关。
用法:CScript脚本名称[脚本选项...][脚本参数...]
选项:
//B批模式:不显示脚本错误及提示信息
//D启用ActiveDebugging
//E:engine使用执行脚本的引擎
//H:CScript将默认的脚本宿主改为CScript.exe
//H:WScript将默认的脚本宿主改为WScript.exe(默认)
//I交互模式(默认,与//B相对)
//Job:xxxx执行一个WSF工作
//Logo显示徽标(默认)
//Nologo不显示徽标:执行时不显示标志
//S为该用户保存当前命令行选项
//T:nn超时设定秒:允许脚本运行的最长时间
//X在调试器中执行脚本
//U用Unicode表示来自控制台的重定向I/O
“脚本名称”是带有扩展名和必需的路径信息的脚本文件名称,如d:\admin\vbscripts\chart.vbs。
“脚本选项和参数”将传递给脚本。脚本参数前面有一个斜杠(/)。每个参数都是可选的;但不能在未指定脚本名称的情况下指定脚本选项。如果未指定参数,则CScript将显示CScript语法和有效的宿主参数。
八、将批处理转化为可执行文件:
由于批处理文件是一种文本文件,任何人都可以对其进行随便编辑,不小心就会把里面的命令破坏掉,所以如果将其转换成.com格式的可执行文件,不仅执行效率会大大提高,而且不会破坏原来的功能,更能将优先级提到最高。Bat2Com就可以完成这个转换工作。
小知识:在DOS环境下,可执行文件的优先级由高到低依次为.com>.exe>.bat>.cmd,即如果在同一目录下存在文件名相同的这四类文件,当只键入文件名时,DOS执行的是name.com,如果需要执行其他三个文件,则必须指定文件的全名,如name.bat。
这是一个只有5.43K大小的免费绿色工具,可以运行在纯DOS或DOS窗口的命令行中,用法:Bat2ComFileName,这样就会在同一目录下生成一个名为FileNme.com的可执行文件,执行的效果和原来的.bat文件一样。
九、时间延迟
什么是时间延迟?顾名思义,就是执行一条命令后延迟一段时间再进行下一条命令。
延迟的应用见下节:“模拟进度条”。
1、利用ping命令延时
例:
@echooff echo延时前:%time% ping/n3127.0.0.1>nul echo延时后:%time% pause
解说:用到了ping命令的“/n”参数,表示要发送多少次请求到指定的ip。本例中要发送3次请求到本机的ip(127.0.0.1)。127.0.0.1可简写为127.1。“>nul”就是屏蔽掉ping命令所显示的内容。
2、利用for命令延时
例:
@echooff echo延时前:%time% for/l%%iin(1,1,5000)doecho%%i>nul echo延时后:%time% pause
解说:原理很简单,就是利用一个计次循环并屏蔽它所显示的内容来达到延时的目的。
3、利用vbs延迟函数,精确度毫秒,误差1000毫秒内
例:
@echooff echo%time% call:delay5000 echo%time% pause exit :delay echoWScript.Sleep%1>delay.vbs CScript//Bdelay.vbs deldelay.vbs goto:eof
运行显示:
10:44:06.45
10:44:11.95
请按任意键继续...
上面的运行结果显示实际延时了5500毫秒,多出来的500毫秒时建立和删除临时文件所耗费的时间。误差在一秒之内。
4、仅用批处理命令实现任意时间延迟,精确度10毫秒,误差50毫秒内
仅用批处理命令就可以实现延迟操作。
例:
@echooff set/pdelay=请输入需延迟的毫秒数: setTotalTime=0 setNowTime=%time% ::读取起始时间,时间格式为:13:01:05.95 echo程序开始时间:%NowTime% :delay_continue set/aminute1=1%NowTime:~3,2%-100 ::读取起始时间的分钟数 set/asecond1=1%NowTime:~-5,2%%NowTime:~-2%0-100000 ::将起始时间的秒数转为毫秒 setNowTime=%time% set/aminute2=1%NowTime:~3,2%-100 ::读取现在时间的分钟数 set/asecond2=1%NowTime:~-5,2%%NowTime:~-2%0-100000 ::将现在时间的秒数转为毫秒 set/aTotalTime+=(%minute2%-%minute1%+60)%%60*60000+%second2%-%second1% if%TotalTime%lss%delay%gotodelay_continue echo程序结束时间:%time% echo设定延迟时间:%delay%毫秒 echo实际延迟时间:%TotalTime%毫秒 pause
运行显示:
请输入需延迟的毫秒数:6000
程序开始时间:15:32:16.37
程序结束时间:15:32:22.37
设定延迟时间:6000毫秒
实际延迟时间:6000毫秒
请按任意键继续...
实现原理:首先设定要延迟的毫秒数,然后用循环累加时间,直到累加时间大于等于延迟时间。
误差:windows系统时间只能精确到10毫秒,所以理论上有可能存在10毫秒误差。
经测试,当延迟时间大于500毫秒时,上面的延迟程序一般不存在误差。当延迟时间小于500毫秒时,可能有几十毫秒误差,为什么?因为延迟程序本身也是有运行时间的,同时系统时间只能精确到10毫秒。
为了方便引用,可将上面的例子改为子程序调用形式:
@echooff echo程序开始时间:%Time% call:delay10 echo实际延迟时间:%totaltime%毫秒 echo程序结束时间:%time% pause exit ::-----------以下为延时子程序-------------------- :delay @echooff if"%1"==""goto:eof setDelayTime=%1 setTotalTime=0 setNowTime=%time% ::读取起始时间,时间格式为:13:01:05.95 :delay_continue set/aminute1=1%NowTime:~3,2%-100 set/asecond1=1%NowTime:~-5,2%%NowTime:~-2%0-100000 setNowTime=%time% set/aminute2=1%NowTime:~3,2%-100 set/asecond2=1%NowTime:~-5,2%%NowTime:~-2%0-100000 set/aTotalTime+=(%minute2%-%minute1%+60)%%60*60000+%second2%-%second1% if%TotalTime%lss%DelayTime%gotodelay_continue goto:eof
十、模拟进度条
下面给出一个模拟进度条的程序。如果将它运用在你自己的程序中,可以使你的程序更漂亮。
@echooff modeconcols=113lines=15&color9f cls echo. echo程序正在初始化... echo. echo┌──────────────────────────────────────┐ set/p=■<nul for/L%%iin(1138)doset/pa=■<nul&ping/n1127.0.0.1>nul echo100%% echo└──────────────────────────────────────┘ pause
解说:“set/pa=■<nul”的意思是:只显示提示信息“■”且不换行,也不需手工输入任何信息,这样可以使每个“■”在同一行逐个输出。“ping/n0127.1>nul”是输出每个“■”的时间间隔,即每隔多少时间输出一个“■”。
十一、特殊字符的输入及应用
开始->运行->输入cmd->edit->ctrl+p(意思是允许输入特殊字符)->按ctrl+a将会显示笑脸图案。
(如果要继续输入特殊字符请再次按ctrl+p,然后ctrl+某个字母)
以上是特殊字符的输入方法,选自[英雄]教程,很管用的。也就是用编辑程序edit输入特殊字符,然后保存为一文本文件,再在windows下打开此文件,复制其中的特殊符号即可。
一些简单的特殊符号可以在dos命令窗口直接输入,并用重定向保存为文本文件。
例:
C:>ECHO^G>temp.txt
“^G”是用Ctrl+G或Alt+007输入,输入多个^G可以产生多声鸣响。
特殊字符的应用也很有意思,这里仅举一例:退格键
退格键表示删除左边的字符,此键不能在文档中正常输入,但可以通过edit编辑程序录入并复制出来。即“”。
利用退格键,可以设计闪烁文字效果
例:文字闪烁
@echooff :start set/p=床前明月光<nul ::显示文字,光标停于行尾 ping-n0127.0.0.1>nul ::设置延迟时间 set/pa=<nul ::输出一些退格符将光标置于该行的最左端(退格符的数量可以自己调整)。 set/pa=<nul ::输出空格将之前输出的文字覆盖掉。 set/pa=<nul ::再次输出退格符将光标置于该行的最左端,这里的退格符数量一定不能比前面的空格数少。 ::否则光标不能退到最左端。 gotostart
例:输出唐诗一首,每行闪动多次
@echooff setlocalenabledelayedexpansion
setstr=床前明月光疑是地上霜举头望明月低头思故乡
::定义字符串str for%%iin(%str%)do( rem由于str中含有空格,则以空格为分隔符将str中的每一个部分依次赋给变量%%i。 setchar=%%i echo. echo. for/l%%jin(0,1,5)do( set/p=!char:~%%j,1!<nul rem依次取出变量char中的每一个字符,并显示。 ping-n0127.0.0.1>nul rem设置输出每个字符的时间延迟。 ) call:hero%%i ) pause>nul exit :hero for/l%%kin(1,1,10)do( ping/n0127.0.0.1>nul set/pa=<nul set/pa=<nul set/pa=<nul ping/n0127.0.0.1>nul set/pa=%1<nul ) ::文字闪动 goto:eof
十二、随机数(%random%)的应用技巧
%RANDOM%系统变量返回0到32767之间的任意十进制数字。由Cmd.exe生成。
2的15次方等于32768,上面的0~32767实际就是15位二进制数的范围。
那么,如何获取100以内的随机数呢?很简单,将%RANDOM%按100进行求余运算即可,见例子。
例:生成5个100以内的随机数
@echooff setlocalenabledelayedexpansion for/L%%iin(115)do( set/arandomNum=!random!%%100 echo随机数:!randomNum! ) pause
运行结果:(每次运行不一样)
随机数:91
随机数:67
随机数:58
随机数:26
随机数:20
请按任意键继续...
求余数运算set/arandomNum=!random!%%100中的100可以是1~32768之间的任意整数。
总结:利用系统变量%random%,求余数运算%%,字符串处理等,可以实现很多随机处理。
思考题目:生成给定位数的随机密码
解答思路:将26个英文字母或10数字以及其它特殊字符组成一个字符串,随机抽取其中的若干字符。
参考答案1:(简单)
@echooff call:randomPassword5pass1pass2 echo%pass1%%pass2% pause exit :randomPassword ::---------生成随机密码 ::---------%1为密码长度,%2及以后为返回变量名称 ::---------for命令最多只能区分31个字段 @echooff setpassword_len=%1 ifnotdefinedpassword_lengoto:eof if%password_len%lss1goto:eof setwordset=abcdefghijklmnopqrstuvwxyz setreturn= setnum=0 :randomPassword1 set/anum+=1 set/anumof=%random%%%26+1 for/f"tokens=%numof%delims="%%iin("%wordset%")dosetreturn=%return%%%i if%num%lss%password_len%gotorandomPassword1 ifnot"%2"==""set%2=%return% shift/2 ifnot"%2"==""gotorandomPassword goto:eof
参考答案2:(最优
@echooff call:randomPassword6pass1pass2pass3 echo%pass1%%pass2%%pass3% pause exit :randomPassword ::---------生成随机密码 ::---------%1为密码长度,%2及以后为返回变量名称 ::---------goto循环、变量嵌套、命令嵌套 @echooff if"%1"==""goto:eof if%1lss1goto:eof setpassword_len=%1 setreturn= setwordset=abcdefghijklmnopqrstuvwxyz023456789_ ::---------------------------循环 :randomPassword1 set/anumof=%random%%%36 callsetreturn=%return%%%wordset:~%numof%,1%% set/apassword_len-=1 if%password_len%gtr0gotorandomPassword1 ::---------------------------循环 ifnot"%2"==""set%2=%return% shift/2 ifnot"%2"==""gotorandomPassword goto:eof
说明:本例涉及到变量嵌套和命令嵌套的应用,见后。
十三、变量嵌套与命令嵌套
和其它编程语言相比,dos功能显得相对简单,要实现比较复杂的功能,需要充分运用各种技巧,变量嵌套与命令嵌套就是此类技巧之一。
先复习一下前面的“字符串截取”的关键内容:
**********************************************
截取功能统一语法格式为:%a:~[m[,n]]%
**********************************************
方括号表示可选,%为变量标识符,a为变量名,不可少,冒号用于分隔变量名和说明部分,符号~可以简单理解为“偏移”即可,m为偏移量(缺省为0),n为截取长度(缺省为全部)。
百分号如果需要当成单一字符,必须写成%%
以上是dos变量处理的通用格式,如果其中的m、n为变量,那么这种情况就是变量嵌套了。
比如设变量word为“abcdefghij”,变量num为“123456789”
%word:~4,1%为e,其中4可以从变量num中取值,即%num:~3,1%,写成组合形式如下:
%word:~%num:~3,1%,1%经测试这种写法不能正确执行,写成%word:~(%num:~3,1%),1%同样不行,那么,怎么实现这种变量嵌套呢?这就必须结合命令嵌套。
什么是命令嵌套呢?简单的说,首先用一条dos命令生成一个字符串,而这个字符串是另一条dos命令,用call语句调用字符串将其执行,从而得到最终结果。
例:用call语句实现命令嵌套
@echooff setstr1=aaaechookbbb echo初始字符串:%str1% echo生成命令字符串如下: echo%str1:~4,7% echo运行命令字符串生成最终结果为: call%str1:~4,7% pause
运行显示:
初始字符串:aaaechookbbb
生成命令字符串如下:
echook
运行命令字符串生成最终结果为:
ok
请按任意键继续...