La réponse sélectionnée fonctionne, mais elle pourrait être améliorée.
- Les options devraient probablement être initialisées à des valeurs par défaut.
- Il serait bon de préserver %0 ainsi que les args %1 et %2 requis.
- Il devient pénible d'avoir un bloc IF pour chaque option, surtout lorsque le nombre d'options augmente.
- Il serait agréable de disposer d'un moyen simple et concis de définir rapidement toutes les options et valeurs par défaut en un seul endroit.
- Il serait bon de supporter des options autonomes qui servent de drapeaux (aucune valeur ne suit l'option).
- On ne sait pas si un arg est entre guillemets. Nous ne savons pas non plus si une valeur arg a été passée en utilisant des caractères échappés. Il est préférable d'accéder à un arg en utilisant %~1 et de mettre l'affectation entre guillemets. Le batch peut alors compter sur l'absence de guillemets, mais les caractères spéciaux restent généralement sûrs sans échappement. (Cette méthode n'est pas infaillible, mais elle permet de gérer la plupart des situations).
Ma solution repose sur la création d'une variable OPTIONS qui définit toutes les options et leurs valeurs par défaut. OPTIONS est également utilisée pour tester si une option fournie est valide. Une quantité considérable de code est économisée en stockant simplement les valeurs des options dans des variables portant le même nom que l'option. La quantité de code est constante quel que soit le nombre d'options définies ; seule la définition de OPTIONS doit être modifiée.
EDITAR - En outre, le code de la :loop doit être modifié si le nombre d'arguments positionnels obligatoires change. Par exemple, il arrive souvent que tous les arguments soient nommés, auquel cas vous souhaitez analyser les arguments commençant à la position 1 au lieu de 3. Ainsi, dans la :loop, les 3 deviennent 1 et les 4 deviennent 2.
@echo off
setlocal enableDelayedExpansion
:: Define the option names along with default values, using a <space>
:: delimiter between options. I'm using some generic option names, but
:: normally each option would have a meaningful name.
::
:: Each option has the format -name:[default]
::
:: The option names are NOT case sensitive.
::
:: Options that have a default value expect the subsequent command line
:: argument to contain the value. If the option is not provided then the
:: option is set to the default. If the default contains spaces, contains
:: special characters, or starts with a colon, then it should be enclosed
:: within double quotes. The default can be undefined by specifying the
:: default as empty quotes "".
:: NOTE - defaults cannot contain * or ? with this solution.
::
:: Options that are specified without any default value are simply flags
:: that are either defined or undefined. All flags start out undefined by
:: default and become defined if the option is supplied.
::
:: The order of the definitions is not important.
::
set "options=-username:/ -option2:"" -option3:"three word default" -flag1: -flag2:"
:: Set the default option values
for %%O in (%options%) do for /f "tokens=1,* delims=:" %%A in ("%%O") do set "%%A=%%~B"
:loop
:: Validate and store the options, one at a time, using a loop.
:: Options start at arg 3 in this example. Each SHIFT is done starting at
:: the first option so required args are preserved.
::
if not "%~3"=="" (
set "test=!options:*%~3:=! "
if "!test!"=="!options! " (
rem No substitution was made so this is an invalid option.
rem Error handling goes here.
rem I will simply echo an error message.
echo Error: Invalid option %~3
) else if "!test:~0,1!"==" " (
rem Set the flag option using the option name.
rem The value doesn't matter, it just needs to be defined.
set "%~3=1"
) else (
rem Set the option value using the option as the name.
rem and the next arg as the value
set "%~3=%~4"
shift /3
)
shift /3
goto :loop
)
:: Now all supplied options are stored in variables whose names are the
:: option names. Missing options have the default value, or are undefined if
:: there is no default.
:: The required args are still available in %1 and %2 (and %0 is also preserved)
:: For this example I will simply echo all the option values,
:: assuming any variable starting with - is an option.
::
set -
:: To get the value of a single parameter, just remember to include the `-`
echo The value of -username is: !-username!
Il n'y a pas vraiment beaucoup de code. La plupart du code ci-dessus est constitué de commentaires. Voici exactement le même code, sans les commentaires.
@echo off
setlocal enableDelayedExpansion
set "options=-username:/ -option2:"" -option3:"three word default" -flag1: -flag2:"
for %%O in (%options%) do for /f "tokens=1,* delims=:" %%A in ("%%O") do set "%%A=%%~B"
:loop
if not "%~3"=="" (
set "test=!options:*%~3:=! "
if "!test!"=="!options! " (
echo Error: Invalid option %~3
) else if "!test:~0,1!"==" " (
set "%~3=1"
) else (
set "%~3=%~4"
shift /3
)
shift /3
goto :loop
)
set -
:: To get the value of a single parameter, just remember to include the `-`
echo The value of -username is: !-username!
Cette solution fournit des arguments de style Unix dans un lot Windows. Ce n'est pas la norme pour Windows - le batch a généralement les options précédant les arguments requis et les options sont préfixées par /
.
Les techniques utilisées dans cette solution sont facilement adaptables à un style d'options Windows.
- La boucle d'analyse syntaxique recherche toujours une option à l'endroit suivant
%1
et cela continue jusqu'à ce que arg 1 ne commence pas par /
- Notez que les affectations SET debe être placé entre guillemets si le nom commence par
/
.
SET /VAR=VALUE
échoue
SET "/VAR=VALUE"
fonctionne. C'est ce que je fais déjà dans ma solution de toute façon.
- Le style standard de Windows exclut la possibilité que la première valeur d'argument requise commence par
/
. Cette limitation peut être éliminée en employant une définition implicite de l'expression //
qui sert de signal pour quitter la boucle d'analyse des options. Rien ne sera stocké pour l'option //
"option".
Mise à jour 2015-12-28 : Soutien aux !
dans les valeurs des options
Dans le code ci-dessus, chaque argument est développé pendant que l'expansion retardée est activée, ce qui signifie que !
sont très probablement dépouillés, ou alors quelque chose comme !var!
est étendu. En outre, ^
peut également être dépouillé si !
est présent. La petite modification suivante apportée au code non commenté supprime la limitation de telle sorte que !
y ^
sont préservés dans les valeurs des options.
@echo off
setlocal enableDelayedExpansion
set "options=-username:/ -option2:"" -option3:"three word default" -flag1: -flag2:"
for %%O in (%options%) do for /f "tokens=1,* delims=:" %%A in ("%%O") do set "%%A=%%~B"
:loop
if not "%~3"=="" (
set "test=!options:*%~3:=! "
if "!test!"=="!options! " (
echo Error: Invalid option %~3
) else if "!test:~0,1!"==" " (
set "%~3=1"
) else (
setlocal disableDelayedExpansion
set "val=%~4"
call :escapeVal
setlocal enableDelayedExpansion
for /f delims^=^ eol^= %%A in ("!val!") do endlocal&endlocal&set "%~3=%%A" !
shift /3
)
shift /3
goto :loop
)
goto :endArgs
:escapeVal
set "val=%val:^=^^%"
set "val=%val:!=^!%"
exit /b
:endArgs
set -
:: To get the value of a single parameter, just remember to include the `-`
echo The value of -username is: !-username!
14 votes
Y a-t-il une raison pour laquelle vous devez faire cela dans un fichier BAT et non en VBScript ? Il y a peut-être un moyen de le faire dans un fichier BAT, mais si vous vous en tenez à cette approche, vous "entrez dans un monde de douleur, fiston" :-)
0 votes
Le mieux est encore d'utiliser PowerShell. Il dispose d'une gestion des paramètres très avancée et intégrée.
2 votes
@AlekDavis, vous avez probablement raison, mais je suis extrêmement phobique du VBScript. Si je me retrouvais un jour à devoir utiliser VBScript, je pense que je serais déjà dans un monde de douleur. Je me contenterai d'écrire un fichier batch pour automatiser quelque chose, et si je veux le rendre plus utile aux gens, j'ajouterai des arguments. Souvent, certains de ces arguments sont facultatifs. Depuis que j'ai quitté le monde de la banque, je n'ai jamais pensé, ou quelqu'un m'a suggéré, que je devais utiliser du VBScript pour cela.
0 votes
@chickeninabiscuit : Le lien ne fonctionne plus. La version de la machine Wayback fonctionne. Je ne sais pas si je modifie ton commentaire pour qu'il ait l'air d'avoir été fait par moi ou non, je vais juste le coller ici à la place :
http://web.archive.org/web/20090403050231/http://www.pcguide.com/vb/showthread.php?t=52323
.