Получить список переданных аргументов в пакетном скрипте Windows (.bat) - PullRequest
360 голосов
/ 10 декабря 2008

Я бы хотел найти пакетный аналог Windows для Bash's $@, который содержит список всех аргументов, переданных в скрипт.

Или я должен беспокоиться о shift?

Ответы [ 12 ]

549 голосов
/ 20 декабря 2008

dancavallaro имеет право, %* для всех параметров командной строки (за исключением самого имени скрипта). Вы также можете найти это полезным:

%0 - команда, используемая для вызова командного файла (может быть foo, ..\foo, c:\bats\foo.bat и т. Д.)
%1 - первый параметр командной строки,
%2 - второй параметр командной строки,
и так далее до %9SHIFT можно использовать для тех, кто после 9-го).

%~nx0 - фактическое имя пакетного файла независимо от вызывающего метода (some-batch.bat)
%~dp0 - диск и путь к скрипту (d: \ scripts)
%~dpnx0 - это полное имя скрипта (d: \ scripts \ some-batch.bat)

Дополнительные примеры в https://www.ss64.com/nt/syntax-args.html и https://www.robvanderwoude.com/parameters.html

132 голосов
/ 10 декабря 2008

%*, кажется, содержит все аргументы, переданные сценарию.

60 голосов
/ 17 марта 2011

%1 ... %n и %* содержат аргументы, но получить к ним доступ может быть сложно, поскольку содержимое будет интерпретировано.
Поэтому невозможно обрабатывать что-то подобное с помощью обычных операторов

myBatch.bat "&"^&

Каждая строка завершается ошибкой, поскольку cmd.exe пытается выполнить один из амперсандов (содержимое% 1 равно "&"&)

set var=%1
set "var=%1"
set var=%~1
set "var=%~1"

Но существует обходной путь с временным файлом

@echo off
SETLOCAL DisableDelayedExpansion

SETLOCAL
for %%a in (1) do (
    set "prompt=$_"
    echo on
    for %%b in (1) do rem * #%1#
    @echo off
) > param.txt
ENDLOCAL

for /F "delims=" %%L in (param.txt) do (
  set "param1=%%L"
)
SETLOCAL EnableDelayedExpansion
set "param1=!param1:*#=!"
set "param1=!param1:~0,-2!"
echo %%1 is '!param1!'

Хитрость заключается в том, чтобы включить echo on и расширить %1 после оператора rem (работает также с% 2 ..% *).
Но чтобы иметь возможность перенаправить вывод echo on, вам нужны два цикла FOR-LOOPS.

Дополнительные символы * # используются для защиты от содержимого, такого как /? (отображает справку для REM).
Или символ ^ в конце строки может работать как многострочный символ.

FOR / F должен работать с отложенным расширением, иначе содержимое с "!" будет уничтожен.
После удаления лишних символов в param1 и вы его получили.

А для безопасного использования param1 включите отложенное расширение.

Редактировать: одно замечание к% 0

%0 содержит команду, используемую для вызова пакета, также сохраняя регистр, как в FoO.BaT
Но после вызова функции %0, а также в %~0 содержится имя функции (или лучше строка, которая использовалась для вызова функции).
Но с %~f0 вы все равно можете вспомнить имя файла.

@echo off
echo main %0, %~0, %~f0
call :myLabel+xyz
exit /b

:MYlabel
echo func %0, %~0, %~f0
exit /b

выход

main test.bat, test.bat, C:\temp\test.bat
func :myLabel+xyz, :myLabel+xyz, C:\temp\test.bat
45 голосов
/ 24 марта 2012

Я обнаружил, что в следующий раз, когда вам нужно будет найти эту информацию. Вместо того, чтобы открывать браузер и гуглить, вы можете просто набрать call /? в вашем cmd, и вы получите его:

...

%* in a batch script refers to all the arguments (e.g. %1 %2 %3
    %4 %5 ...)

Substitution of batch parameters (%n) has been enhanced.  You can
now use the following optional syntax:

    %~1         - expands %1 removing any surrounding quotes (")
    %~f1        - expands %1 to a fully qualified path name
    %~d1        - expands %1 to a drive letter only
    %~p1        - expands %1 to a path only
    %~n1        - expands %1 to a file name only
    %~x1        - expands %1 to a file extension only
    %~s1        - expanded path contains short names only
    %~a1        - expands %1 to file attributes
    %~t1        - expands %1 to date/time of file
    %~z1        - expands %1 to size of file
    %~$PATH:1   - searches the directories listed in the PATH
                   environment variable and expands %1 to the fully
                   qualified name of the first one found.  If the
                   environment variable name is not defined or the
                   file is not found by the search, then this
                   modifier expands to the empty string

The modifiers can be combined to get compound results:

    %~dp1       - expands %1 to a drive letter and path only
    %~nx1       - expands %1 to a file name and extension only
    %~dp$PATH:1 - searches the directories listed in the PATH
                   environment variable for %1 and expands to the
                   drive letter and path of the first one found.
    %~ftza1     - expands %1 to a DIR like output line

In the above examples %1 and PATH can be replaced by other
valid values.  The %~ syntax is terminated by a valid argument
number.  The %~ modifiers may not be used with %*
19 голосов
/ 23 апреля 2013

Способ получения всех аргументов скрипта здесь:

@ECHO off
ECHO The %~nx0 script args are...
for %%I IN (%*) DO ECHO %%I
pause
4 голосов
/ 29 июля 2013

Вот довольно простой способ получить аргументы и установить их как env vars. В этом примере я буду называть их «ключами и значениями».

Сохраните следующий пример кода как "args.bat". Затем вызовите пакетный файл вы сохранили из командной строки. пример: arg.bat --x 90 --y 120

Я предоставил несколько эхо-команд, чтобы пройти через весь процесс. Но конечным результатом является то, что --x будет иметь значение 90, а --y будет иметь значение 120 (то есть, если вы запустите пример, как указано выше ;-)).

Затем вы можете использовать условное выражение «если определено», чтобы определить, следует ли запускать кодовый блок. Итак, скажем запустить: "arg.bat --x hello-world" Затем я мог бы использовать выражение «ЕСЛИ ОПРЕДЕЛЕНО --x echo% - x%», и результатом будет «Привет-мир». Это должно иметь больше смысла, если вы запустите пакет.

@setlocal enableextensions enabledelayedexpansion
@ECHO off
ECHO.
ECHO :::::::::::::::::::::::::: arg.bat example :::::::::::::::::::::::::::::::
ECHO :: By:      User2631477, 2013-07-29                                   ::
ECHO :: Version: 1.0                                                         ::
ECHO :: Purpose: Checks the args passed to the batch.                        ::
ECHO ::                                                                      ::
ECHO :: Start by gathering all the args with the %%* in a for loop.          ::
ECHO ::                                                                      ::
ECHO :: Now we use a 'for' loop to search for our keys which are identified  ::
ECHO :: by the text '--'. The function then sets the --arg ^= to the next    ::
ECHO :: arg. "CALL:Function_GetValue" ^<search for --^> ^<each arg^>         ::
ECHO ::                                                                      ::
ECHO ::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::

ECHO.

ECHO ::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
ECHO :: From the command line you could pass... arg.bat --x 90 --y 220       ::
ECHO ::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
ECHO.
ECHO.Checking Args:"%*"

FOR %%a IN (%*) do (
    CALL:Function_GetValue "--","%%a" 
)

ECHO.
ECHO ::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
ECHO :: Now lets check which args were set to variables...                   ::
ECHO ::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
ECHO.
ECHO ::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
ECHO :: For this we are using the CALL:Function_Show_Defined "--x,--y,--z"   ::
ECHO ::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
ECHO.
CALL:Function_Show_Defined "--x,--y,--z"
endlocal
goto done

:Function_GetValue

REM First we use find string to locate and search for the text.
echo.%~2 | findstr /C:"%~1" 1>nul

REM Next we check the errorlevel return to see if it contains a key or a value
REM and set the appropriate action.

if not errorlevel 1 (
  SET KEY=%~2
) ELSE (
  SET VALUE=%~2
)
IF DEFINED VALUE (
    SET %KEY%=%~2
    ECHO.
    ECHO ::::::::::::::::::::::::: %~0 ::::::::::::::::::::::::::::::
    ECHO :: The KEY:'%KEY%' is now set to the VALUE:'%VALUE%'                     ::
    ECHO :::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
    ECHO.
    ECHO %KEY%=%~2
    ECHO.
    REM It's important to clear the definitions for the key and value in order to
    REM search for the next key value set.
    SET KEY=
    SET VALUE=
)
GOTO:EOF

:Function_Show_Defined 
ECHO.
ECHO ::::::::::::::::::: %~0 ::::::::::::::::::::::::::::::::
ECHO :: Checks which args were defined i.e. %~2
ECHO :::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
ECHO.
SET ARGS=%~1
for %%s in (%ARGS%) DO (
    ECHO.
    ECHO :::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
    ECHO :: For the ARG: '%%s'                         
    IF DEFINED %%s (
        ECHO :: Defined as: '%%s=!%%s!'                                             
    ) else (
        ECHO :: Not Defined '%%s' and thus has no value.
    )
    ECHO :::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
    ECHO.
)
goto:EOF

:done
2 голосов
/ 21 января 2016

Следующий код имитирует массив ('params') - принимает параметры, полученные скриптом, и сохраняет их в переменных params_1 .. params_n, где n = params_0 = число элементов массива:

@echo off

rem Storing the program parameters into the array 'params':
rem Delayed expansion is left disabled in order not to interpret "!" in program parameters' values;
rem however, if a parameter is not quoted, special characters in it (like "^", "&", "|") get interpreted at program launch
set /a count=0
:repeat
    set /a count+=1
    set "params_%count%=%~1"
    shift
    if defined params_%count% (
        goto :repeat
    ) else (
        set /a count-=1
    )    
set /a params_0=count

rem Printing the program parameters stored in the array 'params':
rem After the variables params_1 .. params_n are set with the program parameters' values, delayed expansion can
rem be enabled and "!" are not interpreted in the variables params_1 .. params_n values
setlocal enabledelayedexpansion
    for /l %%i in (1,1,!params_0!) do (
        echo params_%%i: "!params_%%i!"
    )
endlocal

pause
goto :eof
1 голос
/ 27 февраля 2019

enter image description here

Для использования циклов получить все аргументы и в чистом пакетном режиме:

@echo off && setlocal EnableDelayedExpansion

 set "_cnt=0" && for %%Z in (%*) do ( 
 set "_arg_=%%Z" && set /a "_cnt=!_cnt! + 1" && set "_arg_[!_cnt!]=!_arg_!"
 shift && for /l %%l in (!_cnt! 1 !_cnt!) do echo/ The argument n:%%l is: !_arg_[%%l]!
 )

goto :eof 

Ваш код готов что-то сделать с номером аргумента там, где это необходимо, например ...

 @echo off && setlocal EnableDelayedExpansion

 set "_cnt=0" && for %%Z in (%*) do ( 
 set "_arg_=%%Z" && set /a "_cnt=!_cnt! + 1" && set "_arg_[!_cnt!]=!_arg_!"
 shift 

 )

 run_command !_arg_[1]! !_arg_[2]! !_arg_[2]!> log.txt
1 голос
/ 17 августа 2018

Если в кавычках есть параметры, содержащие пробелы, %* не будет работать правильно. Лучшее решение, которое я нашел, это иметь цикл, который объединяет все аргументы: https://serverfault.com/a/22541

set args=%1
shift
:start
if [%1] == [] goto done
set args=%args% %1
shift
goto start

:done
(use %args% here)
0 голосов
/ 31 марта 2016

Много информации, приведенной выше, привело меня к дальнейшим исследованиям и, в конечном итоге, к моему ответу, поэтому я хотел поделиться тем, что я делал, в надежде, что это поможет кому-то еще:

  • Я также хотел передать переменное количество переменных в пакетный файл, чтобы их можно было обработать в этом файле.

  • Я согласился передать их в командный файл, используя кавычки

  • Я бы хотел, чтобы они обрабатывались аналогично приведенному ниже, но с использованием цикла вместо записи вручную:

Итак, я хотел выполнить это:

prog_ZipDeleteFiles.bat "_appPath=C:\Services\Logs\PCAP" "_appFile=PCAP*.?"

И используя магию циклов for, сделайте это в командном файле:

set "_appPath=C:\Services\Logs\PCAP"
set "_appFile=PCAP*.?"

Проблема была в том, что все мои попытки использовать цикл for не работали. Пример ниже:

for /f "tokens* delims= " in %%A (%*) DO (
   set %%A
)

просто сделает:

set "_appPath=C:\Services\Logs\PCAP"

а не:

set "_appPath=C:\Services\Logs\PCAP"
set "_appFile=PCAP*.?"

даже после настройки

SETLOCAL EnableDelayedExpansion

Причина была в том, что цикл for прочитал всю строку и назначил мой второй параметр %% B во время первой итерации цикла. Поскольку% * представляет все аргументы, обрабатывается только одна строка - поэтому происходит только один проход цикла for. Это по замыслу оказывается, и моя логика была неверной.

Поэтому я перестал пытаться использовать цикл for и упростил то, что пытался сделать, используя операторы if, shift и goto. Согласен, это немного взломать, но это больше подходит для моих нужд. Я могу перебрать все аргументы, а затем использовать оператор if, чтобы выйти из цикла, когда обработаю их все.

Заявление о победе за то, что я пытался выполнить:

echo on
:processArguments
:: Process all arguments in the order received
if defined %1 then (
    set %1
    shift
    goto:processArguments
) ELSE (
    echo off 
)

ОБНОВЛЕНИЕ - вместо этого мне пришлось изменить приведенное ниже, я выставлял все переменные среды, пытаясь сослаться на% 1 и используя shift таким образом:

echo on
shift
:processArguments
:: Process all arguments in the order received
if defined %0 then (
    set %0
    shift
    goto:processArguments
) ELSE (
    echo off 
)
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...