从Windows批处理中的文件名字符串获取变量

时间:2020-10-06 06:22:53

标签: windows batch-file cmd forfiles

美好的一天,我有一个包含400个文件的文件夹,我想从文件名中打印一个“名称” 这是结构 ej:

20201323223_vendo.perfil01_17872513294967257_1601950878_live.mp4
20201323223__vvcastrillon_12_17949951031375250_1601939874_live.mp4
2020323123_yessromero.g_17849208194340047_1601945592_live.mp4
2020323223_ziizii_08_17979840166310477_1601929868_live.mp4

我需要的是

vendo.perfil01
_vvcastrillon_12
yessromero.g
ziizii_08

我尝试循环进入文件并分离_并提取2和3令牌数字条件,但结果是错误的并且缺少变量

@echo off
setlocal EnableDelayedExpansion
:loop
SET max=5000
for /F "delims=" %%I in ('dir "*_*_*.mp4" /A-D /B /S 2^>nul ^| %SystemRoot%\System32\findstr.exe /I /L /V "\\outdir\\"') do (for /F "eol=| tokens=2,3 delims=_" %%J in ("%%~nI") do (SET "var="&for /f "delims=0123456789" %%a in ("%%K") do SET var=%%a
if defined var ( set nam=%%J_%%K ) else ( set nam=%%J )
)
echo/!nam!
)
timeout 10 > nul
goto loop

我认为答案是删除_之前的第一个数字,然后删除末尾的字符串_xxxxxx_xxxxxxx_live.mp4,但我不知道如何反向读取令牌

需要任何帮助

3 个答案:

答案 0 :(得分:0)

类似的事情应该有所帮助:

@echo off
setlocal enabledelayedexpansion
for /f "tokens=1* delims=_." %%i in ('dir /b /s /a-d "*_*_*.mp4"2^>nul ^| findstr.exe /ILV "\\outdir\\"') do (
    set "var=%%j"
    for /f "tokens=2,* delims=_." %%a in ("%%j") do echo !var:_%%b=!
)

请记住,使用delims也会分割连续字符,例如双下划线。对于那些需要预先确定具有双下划线的对象,然后让脚本为您添加。

答案 1 :(得分:0)

由于文件名中_甚至相邻_处都有不同数量的delims=_分隔项,因此请使用for /F将它们分成_到特定标记不是最佳选择。

我将改用标准的for-loop,它接收修改的文件名,即每个" """替换并包含在20201323223_vendo.perfil01_17872513294967257_1601950878_live.mp4 中,这导致 SPACE 分隔的部分名称项。所以:

"20201323223" "vendo.perfil01" "17872513294967257" "1601950878" "live.mp4"

更改为:

@echo off
setlocal EnableExtensions DisableDelayedExpansion

rem // Define constants here:
set "_ROOT=%~dp0."           & rem // (full path to target directory)
set "_MASK=*_*_*_*_live.mp4" & rem // (pattern to match file names)
set "_FILT=^[0123456789][0123456789]*_..*_[0123456789][0123456789]*_[0123456789][0123456789]*_live\.mp4$"
                               rem // (additional `findstr` filter for file names)
set /A "_POS=1"              & rem /* (index of first `_`-separated item to extract;
                               rem     `0` is the first item, `-1` is the last) */
set /A "_NUM=-4"             & rem /* (number of `_`-separated items to extract;
                               rem     `-1` means all up to the last item) */

rem // Change into target directory:
pushd "%_ROOT%" && (
    rem // Loop through all matching non-hidden, non-system files:
    for /F "delims= eol=|" %%F in ('
        dir /B /A:-D-H-S "%_MASK%" ^| findstr /I /R /C:"%_FILT%"
    ') do (
        rem // Store current file name, initialise item index, counter and buffer:
        set "FILE=%%F" & set /A "IDX=-1, CNT=-1" & set "BUFF=_"
        rem // Toggle delayed expansion to avoid troubles with `!`:
        setlocal EnableDelayedExpansion
        rem // Count number of items and store one less:
        rem set "TEST=%FILE:_=" & set /A "CNT+=1" & set "TEST=%"
        for %%I in ("!FILE:_=" "!") do set /A "CNT+=1"
        rem // Determine item index position from given index and number:
        if !_POS! lss 0 (set /A "BEG=CNT+_POS+1") else set /A "BEG=_POS"
        if !_NUM! lss 0 (set /A "END=CNT+_NUM+1") else set /A "END=_POS+_NUM-1"
        rem // Transport numbers over `endlocal` barrier:
        for %%C in (!CNT!) do for %%B in (!BEG!) do for %%A in (!END!) do (
            rem // Loop through `_`-separated items of file name:
            for %%I in ("!FILE:_=" "!") do (
                rem // Store current item, increment item index:
                endlocal & set "ITEM=%%~I" & set /A "IDX+=1"
                setlocal EnableDelayedExpansion
                rem // Append current item to buffer if in range:
                if !IDX! geq %%B if !IDX! leq %%A (
                    rem // Transport buffer over `endlocal` barrier:
                    for /F "delims=" %%E in ("BUFF=!BUFF!_!ITEM!") do (
                        endlocal & set "%%E"
                        setlocal EnableDelayedExpansion
                    )
                )
            )
        )
        rem // Return buffer:
        echo(!BUFF:~2!
        endlocal
    )
    rem // Return from target directory:
    popd
)

endlocal
exit /B

在循环之前。在循环中实现一个索引计数器,然后将这些项目附加到缓冲区中,该缓冲区的索引号在一定范围内,该范围取决于项目的总数。

这是一个示例脚本,演示了我的意思:

    {
        "pk": 1,
        "author": {
            "username": "username",
            "email": "email",
            "profile": "image_link"
        }, # AuthorSerializer is worked
        "title": "title1",
        "text": "text1",
        "view": 0,
        # ImageSerializer is not worked
        "like_count": 0,
        # LikerSerializer is not worked
        "comment_count": 0,
        "tag": "hash tag",
        "created_at": "time"
    }

答案 2 :(得分:0)

@ECHO OFF
SETLOCAL
for %%a in (
20201323223_vendo.perfil01_17872513294967257_1601950878_live.mp4
20201323223__vvcastrillon_12_17949951031375250_1601939874_live.mp4
2020323123_yessromero.g_17849208194340047_1601945592_live.mp4
2020323223_ziizii_08_17979840166310477_1601929868_live.mp4
) do  set "filename=%%a" &call :process&echo --------------------------------------
GOTO :EOF

:process
echo stage 1 %filename%
:: step 1 : delete all characters up to and including the first underscore
set "filename=%filename:*_=%"
echo stage 2 %filename%
:: step 2 : find all numeric strings of length 4 or more in remainder
call :strsgt4 %filename:_= %
:: step 3 : replace each numeric string of length 4 or more + preceding underscore with "/" (invalid filename character)
echo stage 3 %filename%
:proc3lp
if "%zapstrings%" neq " " for %%v in (%zapstrings%) do call set "filename=%%filename:_%%v=/%%"&call set "zapstrings=%%zapstrings: %%v=%%"&goto proc3lp 
echo stage 4 %filename%
:: step 5 : Remove all charactersincluding and after the first "/"
for /f "delims=/" %%v in ("%filename%") do echo result %%v
goto :eof

:strsgt4
set "zapstrings= "
:strsgt4loop
set "test=%1"
if not defined test goto :eof
set "test=%test:~4%"
if defined test call :isnum %test%&if not defined notnumber set "zapstrings=%zapstrings% %1"
shift
goto strsgt4loop

:: Determine whether %1 is purely numeric
:isnum
SET "notnumber=9%~1"
FOR /l %%z IN (0,1,9) DO CALL SET "notnumber=%%notnumber:%%z=%%"
GOTO :eof

确实是弄清楚您的规则是什么的问题。

我认为您的规则是:第一个下划线之后的字符串,直到在数字字符串或长度大于3的字符串之前的下一个下划线之前

%%a循环只是将一系列示例字符串提交给:process

内联注释应解释其余部分。