批次-EnableDelayedExpansion,感叹号和文件/文件夹名称

par*_*adf 5 batch-file

我在文件和文件夹名称上使用感叹号()感到困惑。我创建的批处理读取.txt文件的内容,并将列出的文件从文件夹中排序到子文件夹中。
例如:
source.txt有内:file_1.zipfile_2.zip
调用的文件夹systemfile_1.zipfile_2.zipfile_3.zip,等,并批量读取source.txt和这两个文件复制到system/source

我试图避免启用启用DelayedExpansion,但是在添加文件夹浏览器之后,我找不到比使用它更好的方法了。我需要按照显示:listSources:listFolders的方式启用它。 我现在的问题不是对文件夹名称进行太多处理(!)(尽管允许这样做很不错),但是允许文件副本带有(!)

我考虑过逃跑^^!字符或在setlocal和endlocal之间切换,但是我仍在学习如何正确使用它,因此不知道如何处理它。也许有更好的方法?(我不是程序员!)

@echo off
setlocal EnableDelayedExpansion

rem Copy sorted files into destination folder? [Y/N]
set "copyMode=y"
if /i "%copyMode%"=="y" (set appendTitle=[Copy Mode]) else set appendTitle=[Log Mode]

rem Set prefix for output file (used only if copyMode=n)
set prefixMiss=missing_in_

rem Set defaults to launch directory
set rootSource=%~dp0
set rootFolder=%~dp0

:listSources
title Source file selection %appendTitle%
Show folder names in current directory
echo Available sources for sorting:
for %%a in (*.txt) do (
    set /a count+=1
    set mapArray[!count!]=%%a
    echo !count!: %%a
)

:checkSources
set type=source
if not exist !mapArray[1]! cls & echo No source file (.txt) found on current directory^^! & goto SUB_folderBrowser
if !count! gtr 1 set /a browser=!count!+1 & echo.!browser!: Open Folder Browser? & goto whichSource
if !count! equ 1 echo. & set /p "oneSource=Use !mapArray[1]! as source? [Y/N]: "
if /i "%oneSource%"=="y" (set "sourceFile=1" & echo. & goto listFolders) else goto SUB_folderBrowser

:whichSource
echo.
rem Select source file (.txt)
echo Which one is the source file you want to use? Choose !browser! to change directory.
set /p "sourceFile=#: "
if %sourceFile% equ !browser! goto SUB_folderBrowser
if exist !mapArray[%sourceFile%]! echo. & goto listFolders
echo Incorrect input^^! & goto whichSource

:listFolders
title System folder selection %appendTitle%
rem Show folder names in current directory
cd %~dp0
set count=0
echo Available folders for sorting:
for /d %%a in (*) do (
    set /a count+=1
    set mapArray2[!count!]=%%a
    echo !count!: %%a
)

:checkFolders
set type=root
if not exist !mapArray2[1]! cls & echo No folders found on current directory^^! & goto SUB_folderBrowser
if !count! gtr 1 set /a browser=!count!+1 & echo.!browser!: Open Folder Browser? & goto whichSystem
if !count! equ 1 echo. & set /p "oneFolder=Use !mapArray2[1]! as target? [Y/N]: "
if /i "%oneFolder%"=="y" (set "whichSystem=1" & goto whichFolder) else goto SUB_folderBrowser

:whichSystem
echo.
rem Select system folder
echo Which system do you want to sort? Choose !browser! to change directory.
set /p "whichSystem=#: "
if %whichSystem% equ !browser! goto SUB_folderBrowser
if exist !mapArray2[%whichSystem%]! echo. & goto createFolder
echo Incorrect input^^! & goto whichSystem

:createFolder
rem Create destination folder
if /i not "%copyMode%"=="y" goto sortFiles
set destFolder=!mapArray[%sourceFile%]:~0,-4!
if not exist ".\!mapArray2[%whichSystem%]!\%destFolder%" md ".\!mapArray2[%whichSystem%]!\%destFolder%"

:sortFiles
rem Look inside the source file and copy the files to the destination folder
title Sorting files in progress... %appendTitle%
for /f "delims=" %%a in ('type "%rootSource%\!mapArray[%sourceFile%]!"') do (
if exist ".\!mapArray2[%whichSystem%]!\%%a" (
    if /i "%copyMode%"=="y" copy ".\!mapArray2[%whichSystem%]!\%%a" ".\!mapArray2[%whichSystem%]!\%destFolder%\%%~nxa" >nul
    echo %%a
) else (
    echo.
    echo %%a missing & echo.
    if /i not "%copyMode%"=="y" echo %%a >> "%~dp0%prefixMiss%!mapArray2[%whichSystem%]!.txt"
    )
)

title Sorting files finished^^! %appendTitle%

popd & pause >nul & exit

:SUB_folderBrowser
if !count! lss 1 set /p "openBrowser=Open Folder Browser? [Y/N]: "
if !count! lss 1 (
    if not "%openBrowser%"=="y" exit
)
set count=0 & echo Opening Folder Browser... & echo.
if "%type%"=="root" echo Select the root system folder, not the system itself^^!

rem PowerShell-Subroutine to open a Folder Browser
set "psCommand="(new-object -COM 'Shell.Application')^
.BrowseForFolder(0,'Please choose a %type% folder.',0,0).self.path""
for /f "usebackq delims=" %%i in (`powershell %psCommand%`) do set "newRoot=%%i"

if "%type%"=="source" set rootSource=%newRoot%
if "%type%"=="root" set rootFolder=%newRoot%

rem Change working directory
if "%type%"=="source" cls & pushd %rootSource% & goto listSources
if "%type%"=="root" cls & pushd %rootFolder% & echo Selected source file: !mapArray[%sourceFile%]! & echo. & goto listFolders
Run Code Online (Sandbox Code Playgroud)

任何帮助将不胜感激!

dbe*_*ham 3

仅当 FOR 变量包含 时,延迟扩展才会在 FOR 循环中引起问题!

如果从循环内调用 :subroutine,通常可以避免 FOR 循环内的延迟扩展。每次迭代都会重新解析子例程,因此您可以使用正常的%var%变量扩展。但 CALL 会大大减慢速度,所以我想避免它。

有一个简单的解决方案可以避免延迟扩展并避免调用 - 使用 DIR /B 和 FINDSTR /N 将行号注入到目录输出中,FOR /F 可以轻松解析。

例如,以下是 ListSources 循环的解决方案:

for /f "delims=: tokens=1*" %%A in ('dir /b /a-d *.txt^|findstr /n "^"') do (
  set "mapArray[%%A]=%%B"
  set "count=%%A"
  echo %%A: %%B
)
Run Code Online (Sandbox Code Playgroud)

您可以轻松地将相同的技术应用于第二个循环。

您仍然需要在代码的其他非循环部分中进行延迟扩展来扩展数组值。因此,从禁用延迟扩展开始,然后在 :whichSource 中启用它,在第二个循环之前再次禁用它,然后在最后一个循环之后再次启用它。请注意,您不能使用 ENDLOCAL,否则您将丢失之前定义的数组值。因此,当您在禁用和启用延迟扩展之间切换时,您将创建一个小的 SETLOCAL 堆栈。

  • @paradadf - 很好,但你有一个问题。一个 CALL 级别中 SETLOCAL 的最大数量为 32,因此如果迭代次数超过 32 次,则 :sortfiles 中的循环将会失败。通过添加 ENDLOCAL 作为循环的最后一个命令可以解决此问题。这是可以的,因为您没有在循环内设置任何必须保留的变量。 (3认同)