[新手上路]批处理新手入门导读[视频教程]批处理基础视频教程[视频教程]VBS基础视频教程[批处理精品]批处理版照片整理器
[批处理精品]纯批处理备份&还原驱动[批处理精品]CMD命令50条不能说的秘密[在线下载]第三方命令行工具[在线帮助]VBScript / JScript 在线参考
返回列表 发帖

[其他] [50条]不能说的秘密-CMD命令奇诡语法特性汇集

  谨以此文献给我生命中最重要的三个女人,在本文更新期间,是她们给了我很大的物质支持和精神鼓励,支持我最终将此文完成。同时感谢论坛管理层,尤其是批处理之家的namejm,他不仅给我提供了很多线索和建议,同时也在论坛管理上提供了很大的便利和支持。当然,也感谢所有参与讨论并给出建议的朋友们。他们分别是(按发帖时间排序):
BATHOME:neorobin、hanyeguxing、yangfengoo、x9tiancmd、随风、x9tiancmd、Batcher、lxzzr
CN-DOS:qinchun36



声明:
  • 本帖目前已完成初稿,现向全体会员开放阅读,大家可以选择在线阅读或者制作离线阅读版本;
  • 我将不再主要负责本帖的更新,但仍然希望大家能继续跟帖讨论补充,同时麻烦管理层及时更新本帖;
  • 本文代码的运行环境,凡是未明确声明的,均是在Windows XP SP2的CMD下;
  • 此篇可以看做是CMD/DOS下符号的作用参考的姊妹篇,两者内容可能会有重叠;
  • 本文作者不希望各位在此发表无内容的回复,如果只是支持可以选择评分,如果只是路过就请不要留下脚印;
  • 本帖在CNDOS与BATHOME同步更新,互为备份,参考链接如下:
    BATHOME:http://bbs.bathome.net/thread-7629-1-1.html
    CNDOS:http://www.cn-dos.net/forum/viewthread.php?tid=50912

示例:
set varname=var1
set var1=value
call echo %varname%=%%%varname%%%

效果:输出var1=value,实现变量的二次扩展
注释:这种二次扩展的办法在纯DOS中比较常用,CMD下在不能使用变量延迟时也可以考虑

示例:cd.>test.txt
效果:产生一个零字节的文件test.txt
注释:零字节文件的创建,其实有更多的办法(具体见参考链接),但这种目前是最简单可靠的。
链接:http://bbs.bathome.net/thread-939-4-1.html#pid49298
http://www.cn-dos.net/forum/viewthread.php?tid=22336

示例:
md Test Sample
cd Test Sample
rd Test Sample

效果:md和rd命令正常执行,cd命令报错“系统找不到指定的路径。”
注释:md和rd命令隐含支持多个目录的创建和删除,所以把Test Sample理解成了Test和Sample两个目录,
        但cd命令只能是一个目录,而且隐含支持带空格不带引号的路径,所以将Test Sample理解成了一个目录,
        三个命令在认识上的不统一,导致了这个问题的出现。
        另外,不像command.com,cmd下的md和rd已经都支持多级目录的创建和删除了,这个在官方文档已提到。

示例:
cd.>test.html
dir /b *.htm
copy *.htm *.bak

效果:首先列出test.html文件,随后将其扩展名改为.bak
注释:文件名中有通配符且扩展名有3个字节时,无论其扩展名是否有通配符,都会对扩展名进行通配匹配,
        扩展名之后可以添加; , .等特殊字符,不影响通配匹配。
链接:http://www.cn-dos.net/forum/viewthread.php?tid=22336

示例:echo.%var%
效果:如果变量var 为空显示空行,如果不为空则显示其值
注释:echo. 官方文档解释为输出一个空行,用于文本的排版;
但更多的用于防止输出的变量为空时,echo语句出现语法错误;
. 也可以改为, + / : ; [ ] 等特殊字符
链接:http://bbs.bathome.net/thread-4482-1-1.html#pid28940

示例:
echo 文本1 ^
文本2 ^
文本3

效果:将三段文本合并到一行显示
注释:^ 将随后的行结束符取消转义,所以有其它语言中类似续行符的作用
其它命令中也可以这样使用,甚至可以用它截断命令名、文件名或者参数
只是别在引号对中这样使用

示例:set | findstr "\\\\"
效果:显示set命令输出中有一个\ 的行
注释:findstr中不论有没有开启正则,\ 都是元字符的转义字符,而且""对还会将\ 进行再次转义
链接:http://www.cn-dos.net/forum/viewthread.php?tid=21167

示例:
set/p= <nul>示例文字
findstr /a:f1 . 示例文字?
del 示例文字?

效果:显示白色背景黑色前景的“示例文字”
注释:可以多次利用此方法在一行内显示多种不同前景色和背景色的文字
在没有发现这个用途之前,findstr的开关/a可以说是“蛇足”,它在文档中虽有说明,
但语焉不详,更别说如此奇诡的用途;/a: 后的字符可以根据color /?的说明进行调整
链接:http://bbs.bathome.net/thread-1331-1-1.html#pid11864
http://groups.google.com/group/alt.msdos.batch.nt/browse_thread/thread/f819acd4582d5de3/
http://www.cn-dos.net/forum/viewthread.php?tid=35622
http://www.cn-dos.net/forum/viewthread.php?tid=38940

示例:
set/p=不含回车换行符的文本行<nul>test.txt
findstr "$" test.txt
type test.txt|findstr "$"

效果:第二句输出任何文本,第三句输出“不含回车换行符的文本行”
注释:findstr的正则与POSIX正则以及其他标准的正则都不太一样,是个简化版的正则。
        $ 在这里做为正则表达式元字符,匹配行尾的回车和换行符号,而不是真正的行尾。
        而因为测试文本没有回车符和换行符,所以第二句的findstr "$" 匹配失败。
        第三句成功匹配是因为CMD 的“后处理”为文本添加了回车换行并通过管道传给了findstr

示例:
echo 我 你 他>test.txt
findstr "我 你 他" test.txt

效果:没有输出任何文本
注释:finstr在匹配多个中文串时存在问题,使用开关/l或者/r可以得到正确的输出。

示例:
echo.>test.txt && attrib +h test.txt
echo.>test2.txt && attrib +s test2.txt
for %%i in (*.txt) do echo %%i

效果:显示结果不包含带有隐藏属性的test.txt,但包含带有系统属性的test2.txt
注释:不带/a开关的dir 是两者都不显示,for 显然对文件属性有自己的认识

示例:
md TestSample & cd TestSample & cd.>"test 1".txt
for %%f in (*.txt) do echo %%~sf

效果:在WindowsXP SP2环境下显示D:\TESTSA~1\TEST1~1.TXTt,多了一个字符t
注释:短名扩展是基于完整长名路径后的。当路径中既有长目录名,又有含空格的短文件名时,
        CMD的短名扩展不对原长名路径的长度进行修剪,导致短名路径尾部残留原长名路径的字符。
        Windows 2003以上系统无此问题。
链接:http://www.cn-dos.net/forum/viewthread.php?tid=27063

示例:
echo 1 > 1.txt & echo 2 > 2.txt
for %%f in (*.txt) do ren %%f f-%%f

效果:生成两个文件f-2.txt和f-f-1.txt ,1.txt 被改名后再次被改名
注释:在for 语句内使用ren 要谨慎,f-可以改为任意以字母开头的字符串
链接:http://bbs.bathome.net/thread-31727-1-1.html
《批处理BAT脚本中for命令的使用》第8页

示例:for /f "delims=" %%i in (test.txt) do echo %%i
效果:显示test.txt的所有行,除了空行
注释:for/f 的文本遍历会遇到空行时,替代变量%%i被赋值为空,不会执行do的语句块。
        显示空行文本的折中方案是,使用finstr或者find为文本的所有行包括空行添加一个前缀,
        待通过for/f的“防空”保护后,再用set或者tokens把这个前缀去除。如下:

示例:for /f "eol=" %%i in (test.txt) do echo %%i
效果:显示test.txt的文本内容,忽略以起始的行
注释:这个语法之所以称得上“奇诡”,主要是因为官方文档的中文化过程中,
        将“an end of line comment character”翻译成了“行尾字符”(ntcmds.chm)
        在命令行帮助中则是“行注释字符的结尾”,按字面意义理解都会理解成行尾的字符,
        因此误人不浅,实际上应该是“代表行结束的注释字符”,而且这个字符必须在行首。
        不指定eol时,for/f缺省使用分号;作为eol,所以会过滤掉文本中以分号起始的行,
        而"eol="会将双引号指定为eol,使用"eol= delims="则会将空格指定为eol。
        使用示例中的可以近似显示所有的行,因为是一个控制字符,不太可能出现在文本中。

示例:for /l %%i in (2147483635,1,2147483647) do echo %%i
效果:显示两个正数后,将从小到大显示负数,零,正数,负数,依次循环
注释:for /l使用加法,2147483647加一后变为负值,仍然满足循环条件,使之成为无限循环
链接:http://bbs.bathome.net/thread-7659-1-1.html

示例:for /r %f in (.) do @echo %f
效果:分行显示当前目录下所有子目录名,并在行尾添加\.
注释:括号中的句点还可以改为其它不含文件通配符(*,?)的字符或字符串

示例:for /r %f in (test.txt) do @echo.%f
效果:分行显示当前目录下所有子目录名,并在行尾添加\该文件名.当前目录\该文件名
注释:test.txt可以换成不含通配符的其他文件名或者字符串,echo后的.可以换成+、/等特殊字符
链接:http://bbs.bathome.net/viewthread.php?tid=3465&page=1#pid29165

示例:if "〇"=="" echo 〇为空
效果:显示结果为:〇为空
注释:在Windows XP下的936代码页下测试
链接:http://bbs.bathome.net/viewthread.php?tid=7513

示例:if "Z" leq "z" (echo "Z <= z") else echo "Z > z"
效果:显示结果表明Z大于z
注释:if的字符串比较不是按照ASCII或者Unicode码表,而是自有一套规则,详见
链接:http://bbs.bathome.net/thread-6851-1-1.html

示例:
if "e" leq "-" (echo "e <= -") else echo "e > -"
if "ef" leq "-f" (echo "ef <= -f") else echo "ef > -f"

效果:显示结果表明e大于-,而ef小于-f
注释:减号-也可以换成单引号',它们在if的数值比较语句中充当一种另类的转义字符
链接:http://bbs.bathome.net/thread-6853-1-1.html

示例:if 3 gtr -2147483645 (echo 3>-2147483645) else (echo 3<=-2147483645)
效果:显示结果表明3<=-2147483645
注释:if数值比较的核心是减法,依靠对两个数值的减法差值的正负判定大小
而差值变量是32位(DWORD)整型变量,存在-2^31~2^31-1的限定范围
而3- (-2147483645)的差值即超出此限,结果有正转负,判定也由大变小
链接:http://bbs.bathome.net/thread-7659-1-1.html

示例:if defined "test var" (echo pass) else (echo fail)
效果:不管之前有无定义变量test var,显示结果都为fail
注释:if defined判定的变量名包含空格时,不能直接使用 if defined test var echo pass
        此时,test被识别为变量名,var被识别为if的执行体语句块,如果存在变量test,则出现语法错;
        也不能如示例般使用引号对将变量名引起,否则if defined将引号看作变量名的一部分,执行else块
        此时可以使用两个方法使变量名中的空格躲过CMD预处理的词法切分逻辑,详见链接:
        1. for %%f in ("test var") do if defined %%~f (echo pass) else (echo fail)
        2. set varname=test var&if defined !varname! (echo pass) else (echo fail)
        方法2中的!varname!更改为%varname%无效,因为%的变量扩展逻辑在空格的词法切分逻辑之前
        而且变量!varname!不能使用!varname:~1,2!的引用形式,因为逗号影响了CMD预处理的词法切分逻辑
链接:http://bbs.bathome.net/viewthread.php?tid=2050

示例:
md testdir&if exist testdir\nul (echo Pass) else (echo Fail)
md "test dir"&if exist "test dir"\nul (echo Pass) else (echo Fail)

效果:第一句显示Pass,第二句显示Fail
注释:if exist通常用于判断一个路径是否存在,它不能直接分辨出目标路径究竟是目录还是文件。
        而如示例中所用的 if exist test\nul ...,则是从DOS 起广泛使用的判断目录存在的标准方案。
        它借助系统中所存在的aux con nul prn com1 lpt1等标准字符设备名判断目标路径是否指向目录,
        因为这些设备没有可以在任意目录下引用,所以“目录路径\nul”这样的引用被系统认为是有效的,
        但在CMD下,这种方法不适用于判定带引号的目录,这点与if defined "变量名" 的缺陷类似。
        此时,可以使用以下方案:if exist "test dir"\* echo Pass,通配符使

示例:
setlocal enabledelayedexpansion
for /f "delims=" %%a in ('ipconfig^|findstr /i "Address"') do set var=%%a
set CarriageReturn=!var:~-1!

效果:变量CarriageReturn被赋值为一个回车符(0x0d,\r,Cr)
注释:源于 ipconfig 的输出每行行尾都有两个回车符,在被for /f截掉一个后还能剩一个;
        引用时需要用变量延迟的形式,否则就会在预处理中被当做行结束符而被过滤掉
链接:http://bbs.bathome.net/viewthread.php?tid=6692

示例:
reg add hkcu /v test /d "测试" /f
reg query hkcu /v test | find "test"

效果:输出的Test键值丢失了一个字,只余“测”
注释:XP SP3以及以上系统的reg query没有此Unicode带来的问题,解决方法见链接
链接:http://bbs.bathome.net/thread-2046-1-1.html
http://www.cn-dos.net/forum/viewthread.php?tid=21365#pid128521
http://www.cn-dos.net/forum/viewthread.php?tid=22202
http://www.cn-dos.net/forum/viewthread.php?tid=27279#pid182590

示例:
reg add hkcu\test\"quote\"folder /f
reg query hkcu\test\"quote\"folder
reg delete hkcu\test\"quote\"folder

效果:reg添加、查询和删除的注册表键支都是Test,而不是我们设想的Test"quote"folder
注释:XP SP2的reg在注册表键值出现引号" 时可以使用\ 转义,
比如 reg add hkcu /v test /d "\"value\"" /f
但是这个转义字符不能用在注册表键支中,因为它被视作键支层级的分隔符
解决办法是升级reg.exe 为更高版本系统下的reg.exe 外
链接:http://www.bathome.net/viewthread.php?tid=7004

示例:ren test1.txt *t2*
效果:test1.txt将改名为test1.txt2,而非test2.txt
注释:在ren 的任何文件名参数中均可以使用通配符(* 和 ?)。
        如果在 FileName2 中使用通配符,则通配符代表的字符与 FileName1 中的相应字符匹配。
        类似于正则表达式中的最远匹配,但只匹配通配符后的一个字符,其余字符都属于替换字符。
        另外,如果Filename1 含有通配符,那么在NTFS和FAT32下执行会略有差异,详见以下链接。
链接:http://www.cn-dos.net/forum/viewthread.php?tid=29538

示例:md test1 & ren test1 test2
效果:生成test2目录
注释:cmd的ren命令已经悄悄支持目录的改名了。当然,移动目录仍然需要使用move命令。

示例:set/=value
效果:生成变量/,值为value
注释:set将分隔符识别为变量名的一部分,/可以改为( + , . : ; [ \ ] 等特殊字符

示例:
@echo off
set var=
echo.%errorlevel%
set var=value
echo.%errorlevel%

效果:此段代码保存为test.bat执行后分行显示“1 1”,保存为test.cmd执行后显示“1 0”
注释:第一个数字1表示:set试图删除一个不存在的变量var时发生内部错误,产生errorlevel为1
        第二个数字表示:set成功创建变量var后,.cmd文件会重置errorlevel为0,而.bat文件则不会
        也就是说,在.cmd文件中,set成功执行后会重置errorlevel为0,而在.bat文件不管errorlevel
链接:http://www.cn-dos.net/forum/viewthread.php?tid=30968&page=3#pid217157

示例:
set test 1=value
set test 2

效果:test 1的名值对
注释:set 识别单纯的变量名时会丢掉最后一个空格的字符串

示例:set "test=value > 3 "
效果:生成一个变量test,值为"value > 3 "(含空格不含引号)
注释:在变量值涉及特殊字符和尾部空格的时候,用引号对将变量名和变量值括起是个好办法
链接:http://www.cn-dos.net/forum/viewthread.php?tid=17785#pid105897

示例:set "
效果:显示所有的环境变量,其中有包括“=::=”、“=C:=”等变量名含有等号的隐含环境变量
注释:可以把双引号换成一个或多个逗号、双引号、分号或者其混合形式,双引号最好成对使用
        “=::”、“=C:”等变量其代表当前CMD会话曾访问过的各个驱动盘下的当前目录。
        因变量名中包含等号=,正常情况下,set无法对其显示和修改。
        也可能会出现变量“=ExitCode=”,代表CMD所调用外部程序的错误返回码(%Errorlevel%)
链接:http://bbs.bathome.net/thread-7696-1-1.html

        
示例:
set LineFeed=^


效果:变量LineFeed被赋值为一个换行符(0x0a,\n,Lf)
注释:注意set下的两个空行不可少,也就是^后需要三个换行符;
        第一个和第三个换行符被当做行结束标志,第二个被^转义成普通字符;
        引用时需要用变量延迟的形式,否则就会在预处理中被当做行结束符而被过滤掉
链接:http://bbs.bathome.net/viewthread.php?tid=6692

示例:set/a n=1,n=!n!+7
效果:显示n=7,而不是8
注释:!!对的变量延迟语法是针对语句块内的多条语句的,
        而set/a 的逗号表达式是一个语句,而不是语句块,自然不能实现延迟
        解决的办法就是让set/a 自己处理: set/a n=1,n=n+7

示例:
set /a var=_abc
set /a var2=1a

效果:第一句会将变量var赋值为0,第二句会报“无效数字”的语法错,且不会对变量var2赋值
注释:第一句set/a将_abc识别为变量名,因找不到对应变量,所以赋值结果为0;
        第二句set/a将1a识别为操作数,而它又不是一个合法数值,所以提示语法错误,并置errorlevel为9167。
        set/a的判断变量名和操作数的标准类似高级语言:字符串以数字起始是操作数,否则为变量名。
        此外,判断一个字符串是否是合法的十进制数值的方法如下:
        set /a _var=%var% 2>nul
        if "%_var%"=="%var%" echo %var%是合法的十进制数值

示例:set /p test=Text without CrLF(\r\n)<nul
效果:显示一串文本,不含回车换行符
注释:多次使用set /p可以将多串文本输出在一行中,也用在重定向输出过程中

示例:set /p test=<test.txt
效果:将变量test赋值为test.txt的第一行内容
注释:利用了set/p 的重定向输入特性,文本中的回车换行符被cmd理解为语句结束符
        因此可以利用这点切分出文本首行,进而在循环中结合findstr切分出文本的每一行。
        提示:set/p最多可以获取1024个字节作为变量,而环境变量长度限制为8192字节。
链接:https://mp.weixin.qq.com/s/95FNcguB_j22Kfm-l_I-4Q

示例:for /l %%i in (1,1,33) do setlocal
效果:显示“已经达到最大的 setlocal 递归层。”
注释:“使用 setlocal 和 endlocal 命令,可以在 Cmd.exe 的实例中(或在脚本中)进一步将更改局部化。
        Setlocal 创建局部作用范围,而 endlocal 终止局部作用范围。在 setlocal 和 endlocal 作用范围
        内所做的更改将会被放弃,从而保持原始环境不变。这两个命令的嵌套最高可达到 32 级。”
        此段描述引用自官方文档(ntcmds.chm::/ntcmds_shelloverview.htm)
        setlocal意味着重新复制一份环境变量到新的内存地址,而系统只预留了32份地址空间,
        所以环境变量复制份数不能超过32份,也就意味着setlocal的递归次数不能超过32级。
        因此,适时使用endlocal回收不再使用的地址空间不仅是严谨的,也是必要的。

示例:subst ~: C:\Windows & ~:
效果:创建虚拟驱动器~,并跳转到该驱动器下
注释:subst 用于创建、显示和删除虚拟驱动器,该驱动器将指向其它真实或虚拟驱动器文件夹的路径,
        虚拟驱动器的盘符可以是字母、数字以及一些特殊字符,但非字母的盘符不会出现在subst的显示列表中。

示例:type *.txt *.log
效果:显示当前目录下所有文本文件和日志文件的内容
注释:在COMMAND下type是不支持文件通配符的,在CMD下悄悄的支持了

示例:
cd >nul 3>nul
echo Test1
cd >con 4>con
echo Test2

效果:第一句后不再显示以后命令的输出Test1,第二句后恢复显示以后命令的输出Test2
注释:源于重定向后句柄的备份机制,两句中的第一个NUL和CON可以换成其他文件名或者设备名,详见
链接:http://www.cn-dos.net/forum/viewthread.php?tid=16942
http://bbs.bathome.net/viewthread.php?tid=2579

示例:
(
    echo 第一行
    echo 第二行
    echo 第三行
)> test.txt

效果:将圆括号中的所有echo输出重定向到指定的文件test.txt
注释:语句块的重定向使文件句柄的创建或销毁次数减少,因此可以提升执行效率,书写上也更加简单
链接:http://www.cn-dos.net/forum/viewthread.php?tid=46717

示例:
(echo test)|findstr /c:" "

效果:显示test,findstr认为前面的命令输出含有空格
注释:CMD的预处理在分析语句时,会在)和|这些有特殊意义的转义字符前后插入空格,同时剪除多余的空格。
        而当)和|一同出现时,CMD的预处理没有把这些空格从命令行中全部过滤掉,而会遗漏一个空格到echo命令中。
        而echo命令则把这个空格连通之前文本一同输出到管道后命令findstr中,所以导致findstr匹配出含空格行。
链接:http://www.bathome.net/viewthread.php?tid=7629&page=3#pid50356
        http://www.bathome.net/viewthread.php?tid=1480&page=4#pid28166

示例:
>>test.txt echo 行 1
echo>>test.txt 行 2
echo 行>>test.txt 3

效果:将三行输出分别输出到test.txt中
注释:重定向符号可以出现语句中的很多地方,前提是不影响输入输出的正常语法;
        在上面提到语句块的重定向特性出现之前,在纯DOS的环境中行1 的用法更多的被使用,
        因为它使输出语句看起来更加整齐,另外在CMD下也可以避免在行尾出现单独的数字时,
        这个数字可能与重定向符号被一起理解为某个句柄的重定向,而出现意想不到的输出。

示例:
echo.>time.com
time.com

效果:显示COMMAND中TIME命令的输出,并等待用户输入
注释:time可以换成其它COMMAND支持的内部命令名。
对于.COM和没有PE标记的.EXE,Windows都将其认为16位DOS程序,会自动调用其
16位MS-DOS子系统(NTVDM.EXE)解释执行该程序,如果它与COMMAND的内部命令同名,
则优先调用该命令。

示例:
set var=value
:: %var:*=%

效果:显示错误信息“此时不应有 =%”
注释:::虽然被当作注释符号在使用,但是请别忘记它本质上仍然是批处理标签语句,
        cmd在解释执行时也是需要预处理的,虽然只是少量的,但出错也是会提示的。
链接:http://www.cn-dos.net/forum/viewthread.php?tid=40242

示例:
cmd /u /c echo Unicode Text> test.txt & type test.txt
cmd /u /c echo Unicode Text|findstr ".*"
cmd /u /c echo Unicode Text|find /v ""
cmd /u /c echo Unicode Text|more
cmd /u /c echo Unicode Text|sort

效果:前两行命令输出“  n i c o d e   T e x t”的文本,三四行命令将逐字分行输出
        测试文本,并在尾部附五个空行,第五条命令将显示“Unicode”原文。       
注释:cmd /u用于使命令输出或管道中的文本成为Unicode编码,对于纯英文文本,其Unicode
        编码只使用一个高字节,另外一个字节以0x00填补。对于此编码各文本命令支持有很多差异。
        type命令虽然支持显示Unicode文本,但需要文本前缀包含有Unicode的“FFFE”的BOM标记,
        对于cmd /u /c输出的无BOM的Unicode文本将按照ANSI标准输出,字符间的00字节不做转换。
        findstr的特性与type类似,但需要注意的是,它将无法处理重定向输入的Unicode文本。
        而这段文本通过管道送到more和find命令之后,它们会将0x00理解为行结束而输出一个换行符。
        文本后的五个空行,分别是从文本中的回车符、换行符以及它们之间的两个0x00字符转换而来,
        最后一个则是CMD 的“后处理”为文本自动添加的。
        而sort则会将Unicode文本转换为ANSI文本,所以会显示正确的文本,即使其中不包含BOM标记。

示例:mem
效果:首次在CMD窗口中执行会清空窗口,并显示十一个空行,提示符中的长名路径切换为短名;
        第二次执行不会清空窗口,但仍会显示十一个空行;以后执行均只显示一个空行。
        如果在CMD的全屏窗口中执行时,则会正常显示输出结果。
注释:mem.exe是16位DOS 程序,在CMD中调用之前会启用Windows的16位MS-DOS子系统(NTVDM/WOW),
        模拟mem.exe程序类似MSDOS的运行环境,因此导致CMD窗口的输入或输出机制均发生较大的变化,
        首先这个模拟子系统的键盘布局缺省只支持“美式英语的美式键盘”,而大多定制的中文系统已将
        该键盘布局从“文字服务和输入语言”中删除,所以会报“Invalid keyboard code specified”,
        并自动切换回该缺省方案。其次,模拟子系统缺省只支持美式英语的代码页(437),而中文系统下
        的CMD是简体中文的代码页(936),所以会将CMD的活动代码页切换为437,并切换屏幕显示字体,
        因此导致清屏动作,因为以上两个动作是快速发生,所以用户一般不会观察到键盘布局错误的显示。
        如果使用mem>c:\mem.txt重定向命令的输出,则会在输出文本的第一行发现该错误信息。
        在CONFIG.NT中使用COUNTRY命令可以更改模拟DOS环境的国家代码和字符集代码页。
        在AUTOEXEC.NT中使用KB16命令可以更改模拟DOS环境的键盘方案,但不支持中文键盘方案的代码。
链接:http://www.cn-dos.net/forum/viewthread.php?tid=9452
16

评分人数

    • buyiyang: 感谢整理分析技术 + 1
    • 77七: 感谢分享技术 + 1
    • cmd1152: 解开了一些关于批处理的疑惑!技术 + 1
    • 112: 受益匪浅!技术 + 1
    • batchxs: 奇诡技术 + 1

补充一个字符串比较中的问题, 关键是比较时对 短横线 - 和 单引号 ' 的处理问题
  1. if "e" leq "-" (echo 小于) else echo 大于
复制代码
显示的是 大于, 表明 字母 e 是大于短横线 - 的
那么, 字符串 "ef" 也应该是大于 "-f" 的, 可是其实不然
  1. if "ef" leq "-f" (echo 小于) else echo 大于
复制代码
显示的却是 小于
上面所述中, 将短横线换作单引号也得到同样结果, 究竟含有 短横线 或 单引号 的字符串在比较时是如何处理的?
我对此问题的分析和猜测请见: http://bbs.bathome.net/thread-6853-1-1.html
1

评分人数

    • qzwqzw: 感谢提线索,已加入技术 + 1 PB + 10

TOP

在不考虑 4 楼所述 短横线 单引号 在 if 字符串比较中的问题时, if 字符串比较时, ASCII 字符的大小也不是按 ASCII 字符代码值来排序的, 我也从未找到过相关的官方文档说明, 我对其次序作了一个初步的分析: http://bbs.bathome.net/thread-6851-1-1.html
1

评分人数

    • qzwqzw: 感谢参与,已加入技术 + 1 PB + 30

TOP

“〇”算不上特殊字符,但在936代码页(默认),xp sp3系统下,if 会认定该字符为空。
http://bbs.bathome.net/viewthrea ... ight=%2Bhanyeguxing
  1. @echo off
  2. if "〇"=="" (echo 〇为空)else echo 〇不为空
  3. if "12〇34"=="1234" (echo 相同)else echo 不同
  4. if 12〇34==1234 (echo 相同)else echo 不同
  5. pause
复制代码
1

评分人数

    • qzwqzw: 谢谢参与,已加入技术 + 1 PB + 30
寒夜孤星:在没有说明的情况下,本人所有代码均运行在 XP SP3 下 (有问题请发贴,QQ临时会话已关闭)

TOP

我也贡献个
  1. M:\世界之窗>echo.g>ttbb.ttbb
  2. M:\世界之窗>echo.g>ttbb.ttb
  3. M:\世界之窗>echo.g>ttbb.ttbbb
  4. M:\世界之窗>dir /b *.ttb
  5. ttbb.ttb
  6. ttbb.ttbb
  7. ttbb.ttbbb
复制代码
由此可见 cmd 识别扩展名只看前三位。扩展名小于等于3位是完全匹配,大于3位只匹配前三位
1

评分人数

    • qzwqzw: 谢谢参与,鼓励一下技术 + 1 PB + 30

TOP

1,关于变量延迟的特例:
  1. @echo off
  2. set n=1&if defined n echo.定义了变量n,即变量n存在
  3. set/a n+=1,n=n+7
  4. echo.%n%
  5. pause
复制代码
对于 if defined 和 set/a ,不使用%或!括变量,无论是否开了变量延迟,变量本身都会被延迟扩展。如果换成其它写法,就需要开启变量延迟,如:
  1. @echo off&setlocal enabledelayedexpansion
  2. set n=1&if not "!n!"=="" echo.定义了变量n,即变量n存在
  3. set/a n=%n%+1&set/a n=!n!+7
  4. echo.%n%
  5. pause
复制代码
2,关于表达式分隔符,:

以上面的批处理为例,是不可以写成:
  1. @echo off&setlocal enabledelayedexpansion
  2. set n=1&if not "!n!"=="" echo.定义了变量n,即变量n存在
  3. set/a n=%n%+1,n=!n!+7
  4. echo.%n%
  5. pause
复制代码
即用,连起来的表达式,无论是否开了变量延迟,只要这个变量被用!括起来,!n!都不会动态扩展分隔符,前n的变化。
即 set/a n=%n%+1&set/a n=!n!+7 和 set/a n=%n%+1,n=!n!+7 并不等效;
set/a n=n+1,n=n+7set/a n=%n%+1,n=!n!+7 也不等效。
所以:在一个用分隔符 , 连起来的表达式中,!括变量并不能动态扩展该变量在,前的变化,去掉这个!才可以动态扩展该变量在,前的变化。

[ 本帖最后由 hanyeguxing 于 2010-4-17 15:39 编辑 ]
1

评分人数

    • qzwqzw: 谢谢提供,准备加入逗号表达式的部分技术 + 1 PB + 30
寒夜孤星:在没有说明的情况下,本人所有代码均运行在 XP SP3 下 (有问题请发贴,QQ临时会话已关闭)

TOP

回复 6楼 的帖子

变量延迟的特例我需要考虑一下
因为从我的理解
这些语法特性都可以理解
算不上“奇诡”
if defined 和 set /a因为都是在内部处理变量
而不是在预处理过程中处理
也就是说不需要使用变量扩展标记
所以天然具有变量延迟的特性

至于逗号表达式
这是set /a才能处理的特性
cmd的预处理过程
是把两个表达式理解为一句
而不是一个语句块
所以自然不可能延迟扩展变量

TOP

数值比较溢出错误的问题

Winxp CMD 处理的数值范围 是 [-2^31, 2^31-1], 即使两个数值都在此范围内, 但它们进行 if 比较( 除仅仅只是相等或不相等 的比较) 时, 也可能发生溢出错误, 发生情形为两个数的差值的绝对值 大于或等于 2^31 时, 示例
  1. if 3 gtr -2147483645 (echo 3>-2147483645) else (echo 3<=-2147483645)
复制代码
显示结果为:
3<=-2147483645

更多请见: http://bbs.bathome.net/thread-7659-1-1.html
1

评分人数

    • qzwqzw: 谢谢你的提供,准备将if和for的限定分别加入 ...技术 + 1 PB + 30

TOP

原帖由 qzwqzw 于 2010-4-17 17:54 发表
变量延迟的特例我需要考虑一下
因为从我的理解
这些语法特性都可以理解
算不上“奇诡”
if defined 和 set /a因为都是在内部处理变量
而不是在预处理过程中处理
也就是说不需要使用变量扩展标记
所以天然具有变量延迟的特性

至于逗号表达式
这是set /a才能处理的特性
cmd的预处理过程
是把两个表达式理解为一句
而不是一个语句块
所以自然不可能延迟扩展变量
...
这个我同意,当初自学的时候,我绕了好大一个圈子才明白过来
寒夜孤星:在没有说明的情况下,本人所有代码均运行在 XP SP3 下 (有问题请发贴,QQ临时会话已关闭)

TOP

因为帖子字数所限
现在主题帖暂时无法更新
楼上的两位兄弟别着急
我正在想办法
天的白色影子

TOP

  我也来补充几条:
1、
  1. for %i in (*.txt) do echo %i
复制代码
  以上语句不能检测到带隐藏属性的txt文件;
2、
  1. ren 10.pdf *2.c
复制代码
  10.pdf 被改成了什么?10.pdf2.c!具体分析请看这篇文章:http://www.cn-dos.net/forum/viewthread.php?tid=29538
3、
  cmd窗口中运行 for /?,找到一段话:
eol=c           - 指一个行注释字符的结尾(就一个)

  什么意思?是说如果要找到以c字符结尾的某行文字,只需要添加 eol=c 的限制就可以了么?试试运行如下代码:
  1. @echo off
  2. for /f "eol=c" %%i in (test.txt) do echo %%i
  3. pause
复制代码
测试文本test.txt的内容为:
;abc
cab
;cab
acb

  结果如何?居然是不显示第二行内容!
  第二行内容有什么规律?它是以字符 c 打头的。My God,天杀的瘟到死,居然把"忽略以字符c打头的行"给翻译成"忽略以字符c结尾的行"!
4、
  还是上面那个test.txt,代码还是上面那一个,不过去掉了引号内的限制,改成:
  1. @echo off
  2. for /f %%i in (test.txt) do echo %%i
  3. pause
复制代码
  结果,所有以分号打头的行都没显示出来。原来for语句暗含忽略以分号打头的行的功能,因为以分号打头的行在很多语言中都是作为注释语句。
5、
  1. @echo off
  2. for /f %%i in (test.txt) do (
  3.     set str=%%i
  4.     setlocal enabledelayedexpansion
  5.     echo !str!
  6. )
  7. pause
复制代码
  当test.txt内的行数超过32行时,第32行之后,每显示一行的内容,将报错一次:“已经达到最大的 setlocal 递归层。”,当在 echo 的下一句添加 endlocal 语句时,将不会报错。
5、
  1. set /p str=<test.txt
  2. echo %str%
复制代码
  获取 test.txt 第一行的内容,并赋予变量 str

6、
  1. findstr /v "$" test.txt
复制代码
  当 test.txt 最后一行不是空行的时候,将获取最后一行的内容,因为最后一行的字符在结尾处只含回车符号而不含换行符号(貌似是这样解释的吧?),而 $ 在findstr中的准确含义是"匹配回车+换行符号"。

7、
  1. set str=abc
  2. set /a num=%str%&&(echo %str% 是数字) || echo (%str%不是数字)
复制代码
  在SP2环境下,它是可以判断某一字符串是不是纯数字的(要求数字的范围在0~2^32-1这个范围内),但是,到了SP3下之后,这条语句已经不再适用,因为如果某一字符串不是数值,将 set /a num=%str% 之后,num的值将会被赋为0值。

8、
  1. set /a num1=123,num2=345
  2. echo %num1%
  3. echo %num2%
复制代码
  同时给两个变量赋予数值类型的值。如果其中某个变量的值不是数值,在SP2下将出错,在SP3下该变量的值为0。

9、
  1. @echo off
  2. for /f "delims=" %%i in ('type test.txt') do echo "%%i"
  3. pause
复制代码
  对一个追求代码极限简洁的狂热分子来说,type命令可能显得画蛇添足,但是,当你处理的test.txt是Unicode编码的时候,你将体会到什么叫不可或缺。当然,type也可以换成more,不过more在显示长文本的时候会分屏显示。

10、
  1. findstr "我 你" test.txt
复制代码
test.txt中有一行内容:
我 你 他

  上面的代码竟然不能查到这行内容!加上 /i 或 /r 参数,奇迹就会出现。难道这是 findstr 在查找以空格分隔的多个纯中文字符串的时候存在的bug?
1

评分人数

    • qzwqzw: 先把分加上,明天再细看PB + 30 技术 + 1
尺有所短寸有所长,学好批处理没商量;
考虑问题复杂化,解决问题简洁化。

心在天山,身老沧州。

TOP

攒的真多啊
手忙脚乱了半天
才整理的差不多

第5条应该是笔误吧
set /p str=test.txt<nul
好像该是
set /p str=<test.txt

第6条表述有些小问题
那种情况下
最后一行既没有回车符0x0d,\r,也没有换行符0x0a,\n
而"获取最后一行的内容"应该是”不获取最后一行的内容“
改了

第8条没看出跟第7条有什么不同

第9条的内容稍微改了一下加进去了
type支持unicode不算奇诡
cmd/u和type不兼容才算是奇诡
天的白色影子

TOP

印象中还记得有一个关于for /f usebackq 的奇诡语法
不知道谁还能找到?

另外现在帖子是按核心命令行的字母顺序排序
现在的条数越来越多
索引方式就有点乱了
更新起来也比较麻烦
大家有什么其它的好办法没有
最好是多种索引方式并行

[ 本帖最后由 qzwqzw 于 2010-4-19 15:31 编辑 ]
天的白色影子

TOP

原帖由 qzwqzw 于 2010-4-19 14:49 发表
攒的真多啊
手忙脚乱了半天
才整理的差不多

以前碰到过很多诡异的情况,由于不善归纳整理,目前只是自己心知肚明,像楼主这样整理出来与人分享,是件功德无量的事情,值得大家学习。
第5条应该是笔误吧
set /p str=test.txt<nul
好像该是
set /p str=<test.txt

嘿嘿,一激动就写错了,已更改。
第6条表述有些小问题
那种情况下
最后一行既没有回车符0x0d,\r,也没有换行符0x0a,\n
而"获取最后一行的内容"应该是”不获取最后一行的内容“
改了

嗯,最后一行应该是同时没有回车符和换行符。不过我那条代码的执行结果和我的文字描述是吻合的,因为我使用了 /v 来排除回车换行符。
第8条没看出跟第7条有什么不同

第7条的重点是描述 set /a 在不同版本的系统上有不同的表现;而第8条描述是是多个数值赋值语句可以写成一行。
第9条的内容稍微改了一下加进去了
type支持unicode不算奇诡
cmd/u和type不兼容才算是奇诡

呵呵,我的本意是 for 语句不能直接读取Unicode编码的文本,需要借助type或more。
尺有所短寸有所长,学好批处理没商量;
考虑问题复杂化,解决问题简洁化。

心在天山,身老沧州。

TOP

原帖由 qzwqzw 于 2010-4-19 14:50 发表
另外现在帖子是按核心命令行的字母顺序排序
现在的条数越来越多
索引方式就有点乱了
更新起来也比较麻烦
大家有什么其它的好办法没有
最好是多种索引方式并行

  你在顶楼归纳的那4种情况其实就是一个大的分类啊,在这些分类下面,再按照字母顺序排下来就是了。
尺有所短寸有所长,学好批处理没商量;
考虑问题复杂化,解决问题简洁化。

心在天山,身老沧州。

TOP

返回列表