“水滴”批处理脚本 - 包含“&”符号的文件名

7
我正在尝试创建一个批处理文件,可以将其他文件拖放到它上面。具体来说,我正在使用ffmpeg来编辑由手持录音机产生的音频文件。问题是当使用带有“&”符号的文件名时。即使引用输入,任何在“&”之后的内容也会被删除,但仅当文件被拖放到它上面时;如果在命令行上键入文件名输入,则脚本可以正常工作。在cmd窗口关闭之前,我短暂地看到了其余的文件名,并显示错误消息,说明它不被识别为有效命令。 这是我的脚本:
rem 切换到输入文件所在驱动器和目录
%~d1
cd %~p1
rem ffmpeg: 混合到单声道,增加音量 %HOMEDRIVE%%HOMEPATH%\ffmpeg.exe -i "%~nx1" -ac 1 -vol 1024 "%~n1 fixed%~x1" pause
这是在命令行上拖放"ch17&18.mp3"后出现的结果:
C:\Users\computergeeksjw\Desktop>C:\Users\computergeeksjw\ffmpeg.exe -i "ch17" -ac 1 -vol 1024 "ch17 fixed"
[...]
ch17: No such file or directory
如果有影响的话:我正在使用Windows 8开发者预览版。这是导致我的问题吗?在Windows 7或更早版本中是否会出现相同的错误?
2个回答

14

Windows拖放功能中存在一个长期存在的错误,它涉及到包含&^但不包含<空格>的文件路径。

如果文件路径至少包含一个<空格>,则Windows会自动将路径括在引号中,以便正确解析它。如果文件路径包含&^,Windows应该执行相同的操作,但它没有这样做。

如果您创建下面的简单批处理文件并将文件拖入其中,您可以看到问题。

@echo off
setlocal enableDelayedExpansion
echo cmd=!cmdcmdline!
echo %%1="%~1"
pause
exit

!cmdcmdline!变量包含启动批处理文件的实际命令。批处理文件打印出命令行和第一个参数。

如果你拖放一个名为“a.txt”的文件,你会得到:

cmd=cmd /c ""C:\test\drag.bat" C:\test\a.txt"
%1=C:\test\a.txt
Press any key to continue . . .

如果您忽略整个命令周围的引号,就会发现文件参数周围没有引号。因为没有特殊字符,所以没有问题。

现在拖放 "a b.txt",您将会得到:

cmd=cmd /c ""C:\test\drag.bat" "C:\test\a b.txt""
%1="C:\test\a b.txt"
Press any key to continue . . .

你可以看到Windows如何检测名称中的空格并用引号括起文件。再次强调这没有问题。

现在拖放"a&b.txt",你会得到:

cmd=cmd /c ""C:\test\drag.bat" C:\test\a&b.txt"
%1=C:\test\a
Press any key to continue . . .

Windows在文件名中没有找到空格,因此不会用引号括起来。这是个大问题!Windows将"C:\test\a"传递给批处理文件,并将"b.txt"视为在批处理文件完成后要执行的第二个文件。批处理文件中的硬退出命令阻止了任何分割的文件名在批处理之后执行。当然,b.txt永远都不可能被执行。但如果文件名为"a&b.bat",并且存在"b.bat",那么如果批处理文件中没有硬退出命令,就可能会出现问题。

可以将多个文件拖到批处理文件上,每个文件都将作为参数传递。

!cmdcmdline!是唯一可靠访问拖放参数的方法。但如果文件作为普通参数在正常调用批处理文件时传递,它将无法正常工作。

以下是一个批处理文件,它可以检测是否使用拖放方式调用它,而不是普通调用。(它不是绝对可靠的,但我认为它应该在大多数情况下都能正常工作)它将处理每个文件参数,无论调用类型如何,逐个进行处理。(这个过程只是回显文件名,但你可以替换为任何你想要的处理方式。)如果使用拖放方式调用批处理,则它将进行硬退出,以防止分割的文件名出现错误。

@echo off
setlocal disableDelayedExpansion
::
:: first assume normal call, get args from %*
set args=%*
set "dragDrop="
::
:: Now check if drag&drop situation by looking for %0 in !cmdcmdline!
:: if found then set drag&drop flag and get args from !cmdcmdline!
setlocal enableDelayedExpansion
set "cmd=!cmdcmdline!"
set "cmd2=!cmd:*%~f0=!"
if "!cmd2!" neq "!cmd!" (
  set dragDrop=1
  set "args=!cmd2:~0,-1! "
  set "args=!args:* =!"
)
::
:: Process the args
for %%F in (!args!) do (
  if "!!"=="" endlocal & set "dragDrop=%dragDrop%"
  rem ------------------------------------------------
  rem - Your file processing starts here.
  rem - Each file will be processed one at a time
  rem - The file path will be in %%F
  rem -
  echo Process file "%%~F"
  rem -
  rem - Your file processing ends here
  rem -------------------------------------------------
)
::
:: If drag&drop then must do a hard exit to prevent unwanted execution
:: of any split drag&drop filename argument
if defined dragDrop (
  pause
  exit
)

看起来你现有的批处理只能处理一个文件。我不能确定你是否需要对调用进行修改以支持多个文件。我修改了上述批处理程序,使其仅处理第一个参数,并将你的进程替换到参数处理循环中。这还没有经过测试,但我认为它应该适合你。

@echo off
setlocal disableDelayedExpansion
::
:: first assume normal call, get args from %*
set args=%*
set "dragDrop="
::
:: Now check if drag&drop situation by looking for %0 in !cmdcmdline!
:: if found then set drag&drop flag and get args from !cmdcmdline!
setlocal enableDelayedExpansion
set "cmd=!cmdcmdline!"
set "cmd2=!cmd:*%~f0=!"
if "!cmd2!" neq "!cmd!" (
  set dragDrop=1
  set "args=!cmd2:~0,-1! "
  set "args=!args:* =!"
)
::
:: Process the first argument only
for %%F in (!args!) do (
  if "!!"=="" endlocal & set "dragDrop=%dragDrop%"
  rem ------------------------------------------------
  rem - Your file processing starts here.
  rem - Use %%F wherever you would normally use %1
  rem
  rem Change to drive and directory of input file
  %%~dF
  cd %%~pF
  rem ffmpeg: mix to one channel, double the volume
  %HOMEDRIVE%%HOMEPATH%\ffmpeg.exe -i "%%~nxF" -ac 1 -vol 1024 "%%~nF fixed%%~xF"
  rem
  rem - Your file processing ends here
  rem -------------------------------------------------
  goto :continue
)
:continue
if defined dragDrop (
  pause
  exit
)

太棒了!运行得很好。如果不麻烦的话,我想要一份逐行解释脚本如何工作的说明,因为这是我第一次写批处理脚本(尽管我已经有编程经验)。谢谢! - stephenwade

1

我对dbenham在批处理编程方面的技能感到敬佩,默默地仰慕着。 我尝试了他的解决方案,但遇到了两个问题,由于我的声望不够高无法评论,因此在这里提出:

  1. 他的批处理模板第15行最后一个引号前似乎多了一个空格。我认为应该这样写:

      set "args=!cmd2:~0,-1!"
    

    对于不是很擅长批处理编程的人来说,可能会遇到严重的问题,就像我一样。由于愚蠢的“编辑必须至少包含6个字符”的限制,我尝试过但无法编辑dbenham的帖子。

  2. 解决方案通常不适用于完整路径中包含,(逗号)或;(分号)的文件/文件夹。 可以通过在第20行将args括在引号中来修改它,以便在仅有一个文件/文件夹被拖放到批处理文件上时工作:

    for %%F in ("!args!") do (
    

    当拖放超过一个文件/文件夹到批处理文件上时,恐怕没有通用的解决Windows bug的方法可以处理文件路径中的逗号/分号。Windows的SendTo机制显然也存在同样的缺陷(bug),因此不能用于解决拖放错误。因此,微软最终需要修复此bug。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接