我在文件和文件夹名称上使用感叹号(!)感到困惑。我创建的批处理读取.txt文件的内容,并将列出的文件从文件夹中排序到子文件夹中。
例如:
source.txt有内:file_1.zip与file_2.zip
调用的文件夹system有file_1.zip,file_2.zip,file_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)
任何帮助将不胜感激!
仅当 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 堆栈。