web-dev-qa-db-fra.com

Obtenir la liste des arguments passés dans le script batch Windows (.bat)

J'aimerais trouver un équivalent de lot Windows au $@ de Bash qui contient une liste de tous les arguments passés dans un script. 

Ou je dois m'embêter avec shift?

343
wheleph

dancavallaro a le droit, %* pour tous les paramètres de ligne de commande (à l'exception du nom du script lui-même). Vous pourriez aussi trouver cela utile:

%0 - la commande utilisée pour appeler le fichier de commandes (peut être foo, ..\foo, c:\bats\foo.bat, etc.)
%1 est le premier paramètre de la ligne de commande,
%2 est le deuxième paramètre de la ligne de commande,
et ainsi de suite jusqu’au %9 (et SHIFT peut être utilisé pour ceux après le 9) 

%~nx0 - le nom réel du fichier batch, quelle que soit la méthode d'appel utilisée (some-batch.bat)
%~dp0 - lecteur et chemin du script (d:\scripts)
%~dpnx0 - correspond au chemin d'accès complet au script (d:\scripts\some-batch.bat) 

Plus d’informations sur https://www.ss64.com/nt/syntax-args.html et https://www.robvanderwoude.com/parameters.html

535
matt wilkie

%* semble contenir tous les arguments transmis au script.

128
dancavallaro

%1 ... %n et %* contiennent les arguments, mais il peut être difficile d'y accéder, car le contenu sera interprété.
Par conséquent, il est impossible de gérer quelque chose comme ceci avec des déclarations normales

myBatch.bat "&"^&

Chaque ligne échoue, car cmd.exe tente d'exécuter l'un des esperluettes (le contenu de% 1 est "&"&).

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

Mais il existe une solution de contournement avec un fichier temporaire

@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!'

L'astuce consiste à activer echo on et à développer le %1 après une instruction rem (fonctionne également avec% 2 ..% *).
Mais pour pouvoir rediriger la sortie de echo on, vous avez besoin des deux FOR-LOOPS. 

Les caractères supplémentaires * # servent à protéger contre des contenus tels que /? (afficherait l'aide de REM).
Ou un signe ^ à la fin de la ligne pourrait fonctionner comme un caractère multiligne. 

FOR/F doit être un travail avec expansion retardée désactivée, sinon le contenu avec "!" serait détruit.
Après avoir supprimé les caractères supplémentaires dans param1 et vous l’avez obtenu. 

Et pour utiliser param1 de manière sécurisée, activez l’extension différée.

Modifier: une remarque à% 0

%0 contient la commande utilisée pour appeler le lot, tout en préservant la casse comme dans FoO.BaT
Mais après un appel à une fonction %0 et également dans %~0 contient le nom de la fonction (ou mieux, la chaîne qui a été utilisée pour appeler la fonction).
Mais avec %~f0, vous pouvez toujours rappeler le nom du fichier.

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

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

Sortie

main test.bat, test.bat, C:\temp\test.bat
func :myLabel+xyz, :myLabel+xyz, C:\temp\test.bat
60
jeb

J'ai constaté que la prochaine fois que vous devez rechercher ces informations. Au lieu d'ouvrir un navigateur et de le rechercher sur Google, vous pouvez simplement taper call /? dans votre cmd et vous l'obtiendrez:

...

%* 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 %*
43
KFL

La manière de récupérer tous les arguments dans un script est la suivante:

@ECHO off
ECHO The %~nx0 script args are...
for %%I IN (%*) DO ECHO %%I
pause
19
djangofan

Voici un moyen assez simple d’obtenir les arguments et de les définir comme env vars. Dans cet exemple, je les nommerai clés et valeurs.

Enregistrez l'exemple de code suivant sous le nom "args.bat". Appelez ensuite le fichier de commandesvous avez été enregistré à partir d'une ligne de commande . Exemple: arg.bat --x 90 --y 120

J'ai fourni des commandes d'écho pour vous guider tout au long du processus. Mais le résultat final Est que --x aura une valeur de 90 et --y aura une valeur de 120 (c'est-à-dire si vous exécutez l'exemple indiqué ci-dessus ;-)).

Vous pouvez ensuite utiliser l'instruction conditionnelle 'si défini' pour déterminer si vous souhaitez ou non exécuter votre bloc de code . Alors disons: exécutez: "arg.bat --x hello-world" Je pourrais alors utiliser la déclaration "IF DEFINED --x echo% - x%" et les résultats seraient "hello-world". Cela devrait être plus logique si vous exécutez le lot.

@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
3
user2631477

Le code suivant simule un tableau ('params') - prend les paramètres reçus par le script et les stocke dans les variables params_1 .. params_n, où n = params_0 = le nombre d'éléments du tableau:

@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
2
user5280669

Vous pouvez utiliser For Commad, pour obtenir la liste des arguments.

Aide = For /?

Aide = Setlocal /?

Voici ma voie =

@echo off
::For Run Use This = cmd /c ""Args.cmd" Hello USER programming-scientist etc"
setlocal EnableDelayedExpansion
set /a Count=0
for %%I IN (%*) DO (
 Echo Arg_!Count! = %%I
 set /a Count+=1 
)
Echo Count Of Args = !Count!
Endlocal

Vous n'avez pas besoin de passer à la commande !!

0
scientist_7

Beaucoup des informations ci-dessus m'ont amené à approfondir mes recherches et, finalement, à trouver ma réponse. Je souhaitais donc contribuer à ce que je finissais de faire dans l'espoir d'aider quelqu'un d'autre:

  • Je souhaitais également transmettre un nombre varié de variables à un fichier de commandes afin qu'elles puissent être traitées dans le fichier.

  • J'étais d'accord pour les transmettre au fichier de commandes en utilisant des guillemets

  • Je voudrais qu'ils soient traités comme ci-dessous - mais en utilisant une boucle au lieu d'écrire manuellement:

Donc je voulais exécuter ceci:

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

Et via la magie des boucles for, faites ceci dans le fichier batch:

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

Le problème que j'avais était que toutes mes tentatives d'utiliser une boucle for ne fonctionnaient pas. L'exemple ci-dessous:

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

ferais juste:

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

et pas:

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

même après le réglage 

SETLOCAL EnableDelayedExpansion

La raison était que la boucle for lisait la ligne entière et assignait mon deuxième paramètre à %% B lors de la première itération de la boucle. Étant donné que% * représente tous les arguments, il n’ya qu’une seule ligne à traiter - par conséquent, un seul passage de la boucle for a lieu. C'est par conception, il s'avère, et ma logique était fausse.

J'ai donc arrêté d'essayer d'utiliser une boucle for et simplifié ce que j'essayais de faire en utilisant if, shift et une instruction goto. D'accord, c'est un peu un bidouillage mais c'est plus adapté à mes besoins. Je peux passer en revue tous les arguments, puis utiliser une instruction if pour abandonner la boucle une fois que je les ai tous traités.

La déclaration gagnante pour ce que j'essayais d'accomplir:

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

UPDATE - Au lieu de cela, j'ai dû modifier les éléments ci-dessous. J'exposais toutes les variables d'environnement lorsque je tentais de référencer% 1 et d'utiliser shift de cette manière:

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

Si vous avez des paramètres entre guillemets contenant des espaces, %* ne fonctionnera pas correctement ... La meilleure solution que j'ai trouvée est d'avoir une boucle qui joint tous les arguments: 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
Speedstone

 enter image description here

Pour utiliser la mise en boucle, obtenez tous les arguments et en batch pur:

@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 

Votre code est prêt à faire quelque chose avec le numéro d'argument où il faut, comme ...

 @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
0
It was me
 @ echo off 
: start 

 :: Insérez votre code ici 
 echo. %% 1 est maintenant:% ~ 1 
 :: End insérez votre code ici. 
 if "% ~ 2" NEQ "" (
 shift 
 goto: start 
)
0
user298107