标题: [其他] [50条]不能说的秘密-CMD命令奇诡语法特性汇集 [打印本页]
作者: qzwqzw 时间: 2010-4-15 15:01 标题: [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下在不能使用变量延迟时也可以考虑
示例:
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已经都支持多级目录的创建和删除了,这个在官方文档已提到。
示例:
echo 文本1 ^
文本2 ^
文本3
效果:将三段文本合并到一行显示
注释:^ 将随后的行结束符取消转义,所以有其它语言中类似续行符的作用
其它命令中也可以这样使用,甚至可以用它截断命令名、文件名或者参数
只是别在引号对中这样使用
示例:
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 /r %f in (.) do @echo %f
效果:分行显示当前目录下所有子目录名,并在行尾添加\.
注释:括号中的句点还可以改为其它不含文件通配符(*,?)的字符或字符串
示例:
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\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 "
效果:显示所有的环境变量,其中有包括“=::=”、“=C:=”等变量名含有等号的隐含环境变量
注释:可以把双引号换成一个或多个逗号、双引号、分号或者其混合形式,双引号最好成对使用
“=::”、“=C:”等变量其代表当前CMD会话曾访问过的各个驱动盘下的当前目录。
因变量名中包含等号=,正常情况下,set无法对其显示和修改。
也可能会出现变量“=ExitCode=”,代表CMD所调用外部程序的错误返回码(%Errorlevel%)
链接:http://bbs.bathome.net/thread-7696-1-1.html
示例: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可以将多串文本输出在一行中,也用在重定向输出过程中
示例: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下悄悄的支持了
示例:
>>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的内部命令同名,
则优先调用该命令。
示例:
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
作者: neorobin 时间: 2010-4-16 14:47
补充一个字符串比较中的问题, 关键是比较时对 短横线 - 和 单引号 ' 的处理问题- if "e" leq "-" (echo 小于) else echo 大于
复制代码
显示的是 大于, 表明 字母 e 是大于短横线 - 的
那么, 字符串 "ef" 也应该是大于 "-f" 的, 可是其实不然- if "ef" leq "-f" (echo 小于) else echo 大于
复制代码
显示的却是 小于
上面所述中, 将短横线换作单引号也得到同样结果, 究竟含有 短横线 或 单引号 的字符串在比较时是如何处理的?
我对此问题的分析和猜测请见: http://bbs.bathome.net/thread-6853-1-1.html
作者: neorobin 时间: 2010-4-16 15:57
在不考虑 4 楼所述 短横线 单引号 在 if 字符串比较中的问题时, if 字符串比较时, ASCII 字符的大小也不是按 ASCII 字符代码值来排序的, 我也从未找到过相关的官方文档说明, 我对其次序作了一个初步的分析: http://bbs.bathome.net/thread-6851-1-1.html
作者: hanyeguxing 时间: 2010-4-16 18:13
“〇”算不上特殊字符,但在936代码页(默认),xp sp3系统下,if 会认定该字符为空。
http://bbs.bathome.net/viewthrea ... ight=%2Bhanyeguxing- @echo off
- if "〇"=="" (echo 〇为空)else echo 〇不为空
- if "12〇34"=="1234" (echo 相同)else echo 不同
- if 12〇34==1234 (echo 相同)else echo 不同
- pause
复制代码
作者: yangfengoo 时间: 2010-4-16 22:44
我也贡献个- M:\世界之窗>echo.g>ttbb.ttbb
-
- M:\世界之窗>echo.g>ttbb.ttb
-
- M:\世界之窗>echo.g>ttbb.ttbbb
-
- M:\世界之窗>dir /b *.ttb
- ttbb.ttb
- ttbb.ttbb
- ttbb.ttbbb
复制代码
由此可见 cmd 识别扩展名只看前三位。扩展名小于等于3位是完全匹配,大于3位只匹配前三位
作者: hanyeguxing 时间: 2010-4-17 15:11
1,关于变量延迟的特例:- @echo off
- set n=1&if defined n echo.定义了变量n,即变量n存在
- set/a n+=1,n=n+7
- echo.%n%
- pause
复制代码
对于 if defined 和 set/a ,不使用%或!括变量,无论是否开了变量延迟,变量本身都会被延迟扩展。如果换成其它写法,就需要开启变量延迟,如:- @echo off&setlocal enabledelayedexpansion
- set n=1&if not "!n!"=="" echo.定义了变量n,即变量n存在
- set/a n=%n%+1&set/a n=!n!+7
- echo.%n%
- pause
复制代码
2,关于表达式分隔符,:
以上面的批处理为例,是不可以写成:- @echo off&setlocal enabledelayedexpansion
- set n=1&if not "!n!"=="" echo.定义了变量n,即变量n存在
- set/a n=%n%+1,n=!n!+7
- echo.%n%
- 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+7 和 set/a n=%n%+1,n=!n!+7 也不等效。
所以:在一个用分隔符 , 连起来的表达式中,!括变量并不能动态扩展该变量在,前的变化,去掉这个!才可以动态扩展该变量在,前的变化。
[ 本帖最后由 hanyeguxing 于 2010-4-17 15:39 编辑 ]
作者: qzwqzw 时间: 2010-4-17 17:54 标题: 回复 6楼 的帖子
变量延迟的特例我需要考虑一下
因为从我的理解
这些语法特性都可以理解
算不上“奇诡”
if defined 和 set /a因为都是在内部处理变量
而不是在预处理过程中处理
也就是说不需要使用变量扩展标记
所以天然具有变量延迟的特性
至于逗号表达式
这是set /a才能处理的特性
cmd的预处理过程
是把两个表达式理解为一句
而不是一个语句块
所以自然不可能延迟扩展变量
作者: neorobin 时间: 2010-4-17 17:57 标题: 数值比较溢出错误的问题
Winxp CMD 处理的数值范围 是 [-2^31, 2^31-1], 即使两个数值都在此范围内, 但它们进行 if 比较( 除仅仅只是相等或不相等 的比较) 时, 也可能发生溢出错误, 发生情形为两个数的差值的绝对值 大于或等于 2^31 时, 示例- if 3 gtr -2147483645 (echo 3>-2147483645) else (echo 3<=-2147483645)
复制代码
显示结果为:
更多请见: http://bbs.bathome.net/thread-7659-1-1.html
作者: hanyeguxing 时间: 2010-4-17 19:34
原帖由 qzwqzw 于 2010-4-17 17:54 发表
变量延迟的特例我需要考虑一下
因为从我的理解
这些语法特性都可以理解
算不上“奇诡”
if defined 和 set /a因为都是在内部处理变量
而不是在预处理过程中处理
也就是说不需要使用变量扩展标记
所以天然具有变量延迟的特性
至于逗号表达式
这是set /a才能处理的特性
cmd的预处理过程
是把两个表达式理解为一句
而不是一个语句块
所以自然不可能延迟扩展变量
...
这个我同意,当初自学的时候,我绕了好大一个圈子才明白过来
作者: qzwqzw 时间: 2010-4-17 23:05
因为帖子字数所限
现在主题帖暂时无法更新
楼上的两位兄弟别着急
我正在想办法
作者: namejm 时间: 2010-4-18 20:47
我也来补充几条:
1、- for %i in (*.txt) do echo %i
复制代码
以上语句不能检测到带隐藏属性的txt文件;
2、复制代码
10.pdf 被改成了什么?10.pdf2.c!具体分析请看这篇文章:http://www.cn-dos.net/forum/viewthread.php?tid=29538
3、
cmd窗口中运行 for /?,找到一段话:
什么意思?是说如果要找到以c字符结尾的某行文字,只需要添加 eol=c 的限制就可以了么?试试运行如下代码:- @echo off
- for /f "eol=c" %%i in (test.txt) do echo %%i
- pause
复制代码
测试文本test.txt的内容为:
结果如何?居然是不显示第二行内容!
第二行内容有什么规律?它是以字符 c 打头的。My God,天杀的瘟到死,居然把"忽略以字符c打头的行"给翻译成"忽略以字符c结尾的行"!
4、
还是上面那个test.txt,代码还是上面那一个,不过去掉了引号内的限制,改成:- @echo off
- for /f %%i in (test.txt) do echo %%i
- pause
复制代码
结果,所有以分号打头的行都没显示出来。原来for语句暗含忽略以分号打头的行的功能,因为以分号打头的行在很多语言中都是作为注释语句。
5、- @echo off
- for /f %%i in (test.txt) do (
- set str=%%i
- setlocal enabledelayedexpansion
- echo !str!
- )
- pause
复制代码
当test.txt内的行数超过32行时,第32行之后,每显示一行的内容,将报错一次:“已经达到最大的 setlocal 递归层。”,当在 echo 的下一句添加 endlocal 语句时,将不会报错。
5、- set /p str=<test.txt
- echo %str%
复制代码
获取 test.txt 第一行的内容,并赋予变量 str
6、复制代码
当 test.txt 最后一行不是空行的时候,将获取最后一行的内容,因为最后一行的字符在结尾处只含回车符号而不含换行符号(貌似是这样解释的吧?),而 $ 在findstr中的准确含义是"匹配回车+换行符号"。
7、- set str=abc
- set /a num=%str%&&(echo %str% 是数字) || echo (%str%不是数字)
复制代码
在SP2环境下,它是可以判断某一字符串是不是纯数字的(要求数字的范围在0~2^32-1这个范围内),但是,到了SP3下之后,这条语句已经不再适用,因为如果某一字符串不是数值,将 set /a num=%str% 之后,num的值将会被赋为0值。
8、- set /a num1=123,num2=345
- echo %num1%
- echo %num2%
复制代码
同时给两个变量赋予数值类型的值。如果其中某个变量的值不是数值,在SP2下将出错,在SP3下该变量的值为0。
9、- @echo off
- for /f "delims=" %%i in ('type test.txt') do echo "%%i"
- pause
复制代码
对一个追求代码极限简洁的狂热分子来说,type命令可能显得画蛇添足,但是,当你处理的test.txt是Unicode编码的时候,你将体会到什么叫不可或缺。当然,type也可以换成more,不过more在显示长文本的时候会分屏显示。
10、复制代码
test.txt中有一行内容:
上面的代码竟然不能查到这行内容!加上 /i 或 /r 参数,奇迹就会出现。难道这是 findstr 在查找以空格分隔的多个纯中文字符串的时候存在的bug?
作者: qzwqzw 时间: 2010-4-19 14:49
攒的真多啊
手忙脚乱了半天
才整理的差不多
第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不兼容才算是奇诡
作者: qzwqzw 时间: 2010-4-19 14:50
印象中还记得有一个关于for /f usebackq 的奇诡语法
不知道谁还能找到?
另外现在帖子是按核心命令行的字母顺序排序
现在的条数越来越多
索引方式就有点乱了
更新起来也比较麻烦
大家有什么其它的好办法没有
最好是多种索引方式并行
[ 本帖最后由 qzwqzw 于 2010-4-19 15:31 编辑 ]
作者: namejm 时间: 2010-4-19 15:48
原帖由 qzwqzw 于 2010-4-19 14:49 发表
攒的真多啊
手忙脚乱了半天
才整理的差不多
以前碰到过很多诡异的情况,由于不善归纳整理,目前只是自己心知肚明,像楼主这样整理出来与人分享,是件功德无量的事情,值得大家学习。
第5条应该是笔误吧
set /p str=test.txt<nul
好像该是
set /p str=<test.txt
嘿嘿,一激动就写错了,已更改。
第6条表述有些小问题
那种情况下
最后一行既没有回车符0x0d,\r,也没有换行符0x0a,\n
而"获取最后一行的内容"应该是”不获取最后一行的内容“
改了
嗯,最后一行应该是同时没有回车符和换行符。不过我那条代码的执行结果和我的文字描述是吻合的,因为我使用了 /v 来排除回车换行符。
第7条的重点是描述 set /a 在不同版本的系统上有不同的表现;而第8条描述是是多个数值赋值语句可以写成一行。
第9条的内容稍微改了一下加进去了
type支持unicode不算奇诡
cmd/u和type不兼容才算是奇诡
呵呵,我的本意是 for 语句不能直接读取Unicode编码的文本,需要借助type或more。
作者: namejm 时间: 2010-4-19 15:58
原帖由 qzwqzw 于 2010-4-19 14:50 发表
另外现在帖子是按核心命令行的字母顺序排序
现在的条数越来越多
索引方式就有点乱了
更新起来也比较麻烦
大家有什么其它的好办法没有
最好是多种索引方式并行
你在顶楼归纳的那4种情况其实就是一个大的分类啊,在这些分类下面,再按照字母顺序排下来就是了。
作者: qzwqzw 时间: 2010-4-19 16:57 标题: 回复 14楼 的帖子
第一次使用帖子的"引用"
好麻烦
当 test.txt 最后一行不是空行的时候,将获取最后一行的内容
这样的描述很别扭
让我掂不清是最后一行
还是倒数第二行
你在顶楼归纳的那4种情况其实就是一个大的分类啊,在这些分类下面,再按照字母顺序排下来就是了。
我不是专指分类
而是指索引
也就是快速定位的方式方法
比如分楼显示加楼层锚点
正在试for
一头雾水中……
独郁闷不如众郁闷
发上来
也让大家伙费费脑筋
D:\>echo %errorlevel%
0
D:\>(for /f %f in (test) do do set)
系统找不到文件 test。
D:\>echo %errorlevel%
0
D:\>(for /f %f in (test) do do set) 2>nul
D:\>echo %errorlevel%
0
D:\>(for /f %f in (test) do do set) || echo test
系统找不到文件 test。
test
D:\>echo %errorlevel%
5
D:\>(for /f %f in (test) do do set) 2>nul || echo test
test
D:\>echo %errorlevel%
6
D:\>(for /f %f in (uni.txt) do do set) || echo test
test
D:\>echo %errorlevel%
1
D:\>(for /f %f in (uni.txt) do do set) 2>nul || echo test
test
D:\>echo %errorlevel%
1
D:\>(for /f %f in (uni.text) do do set) || echo test
系统找不到文件 uni.text。
test
D:\>echo %errorlevel%
2
D:\>
注释:D盘根存在名为Test的目录
[ 本帖最后由 qzwqzw 于 2010-4-19 17:09 编辑 ]
作者: hanyeguxing 时间: 2010-4-19 16:57
有一个问题算不上诡异,但多数新手却很容易找不到错误的原因:复制代码
这个是很多人经常用的一行代码,但却产生了一个问题。如果在这之前没有定义变量a,则这行代码后会产生%ERRORLEVEL%为1。
例如:- @echo off
- echo.%ERRORLEVEL%
- set a=
- echo.%ERRORLEVEL%
- pause
复制代码
这将直接干扰后面对%ERRORLEVEL%的使用
[ 本帖最后由 hanyeguxing 于 2010-4-20 19:20 编辑 ]
作者: qzwqzw 时间: 2010-4-19 17:02
errorlevel 的细节还很多
足可以单独盖一栋楼了
这是打的基础
http://bbs.bathome.net/thread-7479-1-1.html
作者: qzwqzw 时间: 2010-4-19 17:24 标题: 回复 14楼 的帖子
呵呵,我的本意是 for 语句不能直接读取Unicode编码的文本,需要借助type或more。
得你提醒
又发现一系列”非奇诡“的奇诡
D:\>cmd /u /c echo test123>uni.txt
D:\>more uni.txt
test123
D:\>type uni.txt | more
t
e
s
t
1
2
3
D:\>cmd /u /c echo 测试>uni.txt
D:\>more uni.txt
Km諎
逐字分行又多一个思路
[ 本帖最后由 qzwqzw 于 2010-4-19 17:25 编辑 ]
作者: qzwqzw 时间: 2010-4-19 23:53
set /a 的逗号表达式在命令行帮助和windows帮助中都有提到
大家的代码里也经常用到
而且很多高级语言也支持逗号表达式
所以说明写的简略一点也说得过去
不过你到提醒了我另外一点
做个更新备忘:
set "var=3 > 2 "
作者: qzwqzw 时间: 2010-4-20 16:33 标题: 回复 14楼 的帖子
第7条的重点是描述 set /a 在不同版本的系统上有不同的表现;
关于set/a在xp sp3下的表现我这里与你所述不同
已确认,无论在哪种环境下
set /a var=abc都不会出错
因为它将abc识别为变量名,所以计算结果为0
set /a var=1a都会出错
因为它将1a识别为操作数,所以计算出错
set/a识别变量名和操作数的标准
应改就是是否以字母打头,还有一些其它特殊字符也可以
作者: hanyeguxing 时间: 2010-4-20 16:47
你该用set/a "n=寒夜孤星"去实验,嘿嘿
在sp3下,set/a首先判断是否是数字(第一个字符是否是数字或数字符号),然后判断是否为有效数字
作者: qzwqzw 时间: 2010-4-20 17:03
我已经确认问题细节
你再看一下的我的21楼帖吧
示例:
set /a var=_abc
set /a var2=1a
效果:第一句会将变量var赋值为0,第二句会报“无效数字”的语法错,且不会对变量var2赋值
注释:第一句set/a将_abc识别为变量名,因找不到对应变量,所以赋值结果为0;
第二句set/a将1a识别为操作数,而它又不是一个合法的数值,所以提示语法错误。
set/a的判断变量名和操作数的标准类似高级语言:字符串以数字起始是操作数,否则为变量名。
此外,判断一个字符串是否是合法的十进制数值的方法如下:
set /a _var=%var% 2>nul
if "%_var%"=="%var%" echo %var%是合法的十进制数值
[ 本帖最后由 qzwqzw 于 2010-4-20 17:06 编辑 ]
作者: 随风 时间: 2010-4-20 17:04 标题: 回复 16楼 的帖子
实在没看明白你郁闷的是什么地方
D盘根目录下存在 uni.txt uni.text 这两个文件吗?
如果不存在,那么cmd 返回信息似乎没什么不对的地方呀?
你在最后注释 D盘根目录存在 test 目录是什么意思?for /f 本就不能遍历文件夹吧。
作者: qzwqzw 时间: 2010-4-20 17:09 标题: 回复 24楼 的帖子
测试环境D盘根有文件uni.txt和目录test,没有uni.text
请注意几种情况下不同的errorlevel
我们以为是相同的错误情况
结果出现了不同的errorlevel
作者: 随风 时间: 2010-4-20 17:23
对cmd这些底层的东西向来是敬而远之~
qzwqzw 的功力真是令人羡慕,赞一个先。
下面有几个应该算不上是奇诡,但也有些特殊之处
if defined ab cd 无法检测含空格的变量名是否定义。
相关讨论 http://bbs.bathome.net/viewthread.php?tid=2050&highlight=defined
一种比较另类的变量截取
http://bbs.bathome.net/viewthrea ... ge=1&extra=page%3D4
楼主在13说的 关于for /f usebackq 的奇诡语法 在这里
http://bbs.bathome.net/viewthrea ... hlight=%C1%E9%D2%EC
[ 本帖最后由 随风 于 2010-4-20 17:29 编辑 ]
作者: hanyeguxing 时间: 2010-4-20 17:43
原帖由 qzwqzw 于 2010-4-19 17:24 发表
得你提醒
又发现一系列”非奇诡“的奇诡
逐字分行又多一个思路
实际应用中受到很大的限制
1,对于ANSI编码的,必须使用cmd /u去转换
2,ANSI编码的文本内要求只能有字母、英文符号和数字
不过想到个简单的显示20个空行的方法,哈哈:- fsutil file createnew uni.txt 20
- type uni.txt | more
复制代码
[ 本帖最后由 hanyeguxing 于 2010-4-20 18:02 编辑 ]
作者: qzwqzw 时间: 2010-4-20 19:24 标题: 回复 27楼 的帖子
晕
简单问题复杂化
显示20个空行
这样不更简单吗
for /l %i in (1,1,20) do echo.
逐字分行的意义
不在于分行显示
而在于分行处理
比如下面的代码- for /f %f in ('cmd /u /c echo test^|more') do @echo value=%f
复制代码
在这个意义下
我们通常处理的是单个字符串变量
那么西方字符集的限制就不是什么大问题
至于cmd /u
闲着也是闲着
干嘛不用呢?
[ 本帖最后由 qzwqzw 于 2010-4-20 19:31 编辑 ]
作者: hanyeguxing 时间: 2010-4-20 19:30
问题在于我就是通过这个才认清more逐字分行的原因啊,嘿嘿
作者: 随风 时间: 2010-4-20 20:13
又发现一个奇诡的问题
以下代码,不管是否存在a.txt都无法正确运行,把rem 换成其它的命令则可以,或去掉最外面的一对括弧也可以。- @echo off
- (for /f %%f in (a.txt) do echo.123&rem abc)
- pause
复制代码
作者: yangfengoo 时间: 2010-4-20 20:19
我记得 dir *b*.txt
用*x*来匹配含X的文件名好像有BUG 有时会列出一些不含X的文件名
具体例子不记得了
作者: hanyeguxing 时间: 2010-4-20 20:30
原帖由 随风 于 2010-4-20 20:13 发表
又发现一个奇诡的问题
以下代码,不管是否存在a.txt都无法正确运行,把rem 换成其它的命令则可以,或去掉最外面的一对括弧也可以。@echo off
(for /f %%f in (a.txt) do echo.123&rem abc)
pause
因为)成了rem的一部分,所以for前的(就孤立了
例如:- for /f %%f in (a.txt) do echo.123&rem abc&echo.这个不显示
复制代码
[ 本帖最后由 hanyeguxing 于 2010-4-20 20:33 编辑 ]
作者: 随风 时间: 2010-4-20 20:35 标题: 回复 32楼 的帖子
是哦
作者: Batcher 时间: 2010-4-20 21:30
复制代码
不能直接判断C:\a.txt是文件还是文件夹,这个算吗?
作者: Batcher 时间: 2010-4-20 21:36
复制代码
XP系统里面行尾会多出一个CR,Vista、Win7系统里面没有这个问题,这个算不?
作者: qzwqzw 时间: 2010-4-20 21:43
34楼这个按说不算
if exist没有义务为你判断是文件还是目录
不过可以用以下两个办法判断目录
if exist %1\. echo %1是目录
if exist %1\nul echo %1是目录
这两个可以算
35楼内容之前就已经添加了
搜索一下主题的ipconfig
作者: Batcher 时间: 2010-4-20 21:51
复制代码
要不要注明一下这个方法最多只能读取1024字节?
作者: neorobin 时间: 2010-4-20 21:59
双引号, 空格对 set 输出和变量识别的影响
详见 http://bbs.bathome.net/thread-7696-1-1.html
一. 半角双引号, 半角空格, 全角空格都可能导致变量名不能被 set 正常识别(其中部分原因和下面第二条有关)
二. 以下情形可以显示每个驱动器下的最后工作目录(排在空 set 命令显示的环境变量信息之前):
1. set 后面是一个或若干个全角空格(可以紧跟)
2. set 后面是一个双引号(不能紧跟) 或 一对半角双引号 " (一对双引号中间可以含有若干个半角或全角空格)
作者: Batcher 时间: 2010-4-20 22:01
是不是需要把findstr的最后行问题在补充的详细一些?
http://www.bathome.net/viewthrea ... amp;page=2#pid32098
要不要介绍一个这个多余的空格?
http://www.bathome.net/viewthrea ... amp;page=4#pid28166
http://www.bathome.net/viewthrea ... amp;page=2#pid35901
作者: lxzzr 时间: 2010-4-20 23:15 标题: 也来添加个
代码:
-
- @ECHO OFF
- FOR /F "DELIMS=" %%A IN ('MORE +7^<%~FS0') DO ECHO %%A
- PAUSE>NUL
- COPY %0 TEST.TXT>NUL
- MORE +7<"TEST.TXT"
- PAUSE>NUL
- EXIT
- ;忽略1
-
- ;忽略2
-
- 忽略 ";" 开头的行和空行
-
复制代码
尚不清楚是FOR命令还是MORE,还是管道符的作用....
作者: neorobin 时间: 2010-4-20 23:22
新发现半角逗号 "," 也对 set 命令有影响, 用逗号作参数也可 set, 来输出各驱动器的最后工作目录, 对 set 命令变量识别的影响也和 38楼所述相同或相似
set, 可以显示工作目录, 让人想起了 dos 下, 逗号可以加强 dir 命令
[ 本帖最后由 neorobin 于 2010-4-20 23:26 编辑 ]
作者: hanyeguxing 时间: 2010-4-20 23:25
原帖由 lxzzr 于 2010-4-20 23:15 发表
代码:
@ECHO OFF
FOR /F "DELIMS=" %%A IN ('MORE +7^<%~FS0') DO ECHO %%A
PAUSE>NUL
COPY %0 TEST.TXT>NUL
MORE +7<"TEST.TXT"
PAUSE>NUL
EXIT
;忽略1
;忽略2
忽略 ";" 开头的行和空行
尚不清楚是FOR命令还是MORE,还是管道符的作用.. ...
for的作用,例如,你在for里强制eol= 为空格,就可以看到;的内容了。
(俺错了,偷偷改过,嘿嘿)
[ 本帖最后由 hanyeguxing 于 2010-4-20 23:36 编辑 ]
作者: qzwqzw 时间: 2010-4-21 10:34 标题: 回复 39楼 的帖子
多余空格的问题已基本确认
应该是CMD预处理括号对()和管道符时插入的
请看以下示例
D:\>(echo.abc) > 1.txt
D:\>echo.abc|more>2.txt
D:\>(echo.abc)|more>3.txt
只有3.txt有空格
而1.txt,2.txt都没有
这说明CMD的预处理在做语句的词法切分时
为了方便以后的预处理识别)和|这些转义字符
而特意在这些字符前后插入一个空格
同时也会把多余的空格去除
这种增减空格的现象
在不含echo off的批处理中可以看到很多
比如以下的用法
- (echo test)
- (echo.test)
- echo test|more
- echo.test|more
- (echo test)> a.txt
- (echo.test) > a.txt
- echo.test|more>b.txt
- (echo.test)|more>c.txt
复制代码
D:\>(echo test )
test
D:\>(echo.test)
test
D:\>echo test | more
test
D:\>echo.test | more
test
D:\>(echo test ) 1>a.txt
D:\>(echo.test) 1>a.txt
D:\>echo.test | more1>b.txt
D:\>(echo.test) | more1>c.txt
[ 本帖最后由 qzwqzw 于 2010-4-21 10:39 编辑 ]
作者: Batcher 时间: 2010-4-27 17:54
顶楼内容里面的findstr /l不一定能够解决汉字“或”的问题,测试环境:XP SP3 Pro CS,测试结果:
C:\Test>type a.txt
我们
他们
123
C:\Test>findstr /l "我们 他们" a.txt
C:\Test>findstr /r "我们 他们" a.txt
我们
他们
C:\Test>findstr "我们.* 他们" a.txt
我们
他们
作者: qzwqzw 时间: 2010-4-28 14:39
findstr 匹配多个中文串的问题比较棘手
目前找不到问题的实质原因
所以我在顶楼也只是含糊其辞
如果有更多的测试和证据
足以让人判断其问题的源头
那么我可以考虑更新一下
作者: hanyeguxing 时间: 2010-5-2 02:08
- @echo off
- set m=1234567890
- echo.测试1:%m:~1,1%
- echo.测试2:%m:~20,1%
- echo.测试3:%m:~20%
- echo.测试4:%m:~8,-5%
- echo.测试5:%m:~0,1%
- echo.测试6:%m:~-0,1%
- echo.测试7:%m:~-20,1%
- echo.测试8:%m:~-5%
- echo.测试9:%m:~-20%
- pause
复制代码
在 XP SP3 Pro 下,显示结果:
测试1:2
测试2:
测试3:
测试4:
测试5:1
测试6:1
测试7:1
测试8:67890
测试9:1234567890
测试1、测试5、测试6、测试8,都是显示正常。
测试2、测试3,因为超出了范围,所以为空。
字符串都是从左向右截取的,所以测试4为空。
当数值为负,且超出范围时,是否也为空呢?从测试7和测试9中可以看出,显然不是。
形如%m:~-20,1%,超出范围时,依然截取并获取了最左边的字符。
形如%m:~-20%,超出范围时,总是截取并获取整个字符串。
所以:在截取字符串时,即使负的数值超出范围,依然是从最左边的字符开始截取。
[ 本帖最后由 hanyeguxing 于 2010-5-2 02:13 编辑 ]
作者: namejm 时间: 2010-5-2 03:20
原帖由 Batcher 于 2010-4-27 17:54 发表
顶楼内容里面的findstr /l不一定能够解决汉字“或”的问题,测试环境:XP SP3 Pro CS,测试结果:
C:\Test>type a.txt
我们
他们
123
C:\Test>findstr /l "我们 他们" a.txt
C:\Test>findstr /r "我们 他们" a.txt
我们
他们
C:\Test>findstr "我们.* 他们" a.txt
我们
他们
请注意,你这里使用的是 /l 而不是 /i,前者是按字符搜索的意思,后者是不区分大小写的意思。
作者: Batcher 时间: 2010-5-2 09:30 标题: 回复 47楼 的帖子
我举得例子是针对楼主的这句:
finstr在匹配多个中文串时存在问题,使用开关/l或者/r可以得到正确的输出。
楼主在这里没有提到/i啊
作者: qzwqzw 时间: 2010-5-2 17:38 标题: 回复 46楼 的帖子
早就习惯了变量的这种扩展形式
忘了这也是不符合大多数人的认知习惯的
其实关于增强的变量扩展方面
负值的使用也在挑战大多数人的认知
使用正值就是起始的偏移量+长度
使用负值却不是起始的反向偏移量+反向长度
而是起始的反向偏移量+终止的反向偏移量
我最初在理解set的这句说明了
也还纳闷了老半天
为什么呢?
这大概说明了
增强的变量扩展在对正值和负值的处理上
采用了完全不同的两种处理逻辑
所以正值超出后为空
而负值超出后为首字符地址的逻辑
也就不难理解了
作者: hanyeguxing 时间: 2010-5-6 13:36
1,dir的 /l 参数是按小写字母显示未排序的目录名和文件名。/l 不会将扩展字符转换为小写字符。
2,dir的 /s 参数是列出指定目录及所有子目录中出现的每个指定的文件名。
3,%%~nxi是扩展%%i的文件名和扩展名的。(%%~ni、%%~fi等也是这样)
当这三个复合到一起,就诡异了:- @echo off
- for /f "delims=" %%i in ('dir /s/l/b/a "D:\rec\测试"') do echo.%%i和%%~nxi
- pause
复制代码
%%i是小写的,%%~nxi却是原始的大小写
而当不使用 /s 时,%%~nx和%%i一样,都转成小写。
[ 本帖最后由 hanyeguxing 于 2010-5-6 13:57 编辑 ]
作者: qzwqzw 时间: 2010-5-6 18:16 标题: 回复 50楼 的帖子
这是可以理解的
%%i是dir的原样输出
所以受到dir/l的控制
而%%~nxi是扩展输出
是受到变量扩展逻辑控制的
dir/s/b将给出文件的完整路径
而在%%~nxi扩展时能根据路径找到文件
自然会把变量当作文件名处理
所以是文件自身的大小写状态
没有/s将只给出文件名
而如果当前路径不是dir的路径
则%%~nxi没有办法找到对应文件
自然只有按“当前路径+替代变量“的形式扩展
所以是替代变量%%i原来的大小写状态
作者: hanyeguxing 时间: 2010-5-6 20:02 标题: 回复 51楼 的帖子
- @echo off
- set "r=D:\rec\abc.txt"
- for %%a in (0 1) do call:han %r% %%a
- set "r=abc.txt"
- for %%a in (0 2) do call:han %r% %%a
- pause&exit
- :han
- if %~2==1 cd.>D:\rec\ABC.txt
- if %~2==2 cd.>ABC.txt
- echo.参数%1和%~nx1与%~f1
- for %%i in (%~1) do echo.变量%%i和%%~nxi与%%~fi
复制代码
写了个脚本测试,发现参数和变量的替换扩展确实如是按如此规律此显示的。
%%i 的大小写由 (set) 直接控制,但 %%~nxi 则受 for 能否在扩展中找到文件的控制。如果扩展不到该文件,则按 %%i 即 (set) 中的赋值大小写为准;如果能够扩展到该文件,则按实际文件的大小写为准。 %%~i 例外,原因是 %%~i 直接删除 %%i 的前后引号 ("") ,而不扩展寻找文件。
参数的增强扩展同样如此。
[ 本帖最后由 hanyeguxing 于 2010-5-6 20:49 编辑 ]
作者: 2013haohaoma123 时间: 2013-7-26 12:22
md .\%input%是什么意思
作者: 网上邻居 时间: 2016-3-17 17:35
太多深奥的东西了,诡异的预处理,甚至微软本身的一些奇怪逻辑,批处理是个大坑。
作者: LJY4.0 时间: 2022-6-12 13:17
回复 1# qzwqzw
帮助很大
作者: Zhanggli 时间: 2023-9-1 09:16
华山论剑,高手过招够精彩
欢迎光临 批处理之家 (http://www.bathome.net/) |
Powered by Discuz! 7.2 |