चयनित उत्तर काम करता है, लेकिन यह कुछ सुधार का उपयोग कर सकता है।
- विकल्पों को संभवत: डिफ़ॉल्ट मानों के लिए प्रारंभ किया जाना चाहिए।
- % 0 और साथ ही आवश्यक आर्ग्स% 1 और% 2 को संरक्षित करना अच्छा होगा।
- हर विकल्प के लिए एक IF ब्लॉक होना एक दर्द बन जाता है, खासकर जब विकल्प की संख्या बढ़ती है।
- एक जगह पर सभी विकल्पों और चूक को जल्दी से परिभाषित करने के लिए एक सरल और संक्षिप्त तरीका रखना अच्छा होगा।
- झंडे के रूप में काम करने वाले स्टैंड-अलोन विकल्पों का समर्थन करना अच्छा होगा (विकल्प के बाद कोई मूल्य नहीं)।
- हम नहीं जानते कि क्या एक arg उद्धरण में संलग्न है। और न ही हम जानते हैं कि बच गए वर्णों का उपयोग करके एक arg मान पारित किया गया था या नहीं। % ~ 1 का उपयोग करके एक arg तक पहुँचने के लिए बेहतर है और उद्धरण के भीतर असाइनमेंट संलग्न करें। फिर बैच घेरने वाले उद्धरणों की अनुपस्थिति पर भरोसा कर सकता है, लेकिन विशेष अक्षर अभी भी बच के बिना सुरक्षित हैं। (यह बुलेट प्रूफ नहीं है, लेकिन यह ज्यादातर स्थितियों को संभालता है)
मेरा समाधान एक विकल्प चर के निर्माण पर निर्भर करता है जो सभी विकल्पों और उनकी चूक को परिभाषित करता है। विकल्प का उपयोग यह जांचने के लिए भी किया जाता है कि क्या आपूर्ति का विकल्प मान्य है। कोड की एक जबरदस्त मात्रा को विकल्प के रूप में समान नाम वाले चर में विकल्प मानों को संग्रहीत करके बचाया जाता है। कितने विकल्पों को परिभाषित किया गया है, इसकी परवाह किए बिना कोड की मात्रा स्थिर है; केवल विकल्प की परिभाषा को बदलना होगा।
EDIT - इसके अलावा, यदि अनिवार्य स्थिति संबंधी तर्कों की संख्या में परिवर्तन होता है, तो लूप कोड बदलना होगा। उदाहरण के लिए, अक्सर सभी तर्कों का नाम लिया जाता है, जिस स्थिति में आप 3 के बजाय 1 से शुरू होने वाले तर्कों को पार्स करना चाहते हैं। इसलिए: लूप के भीतर, सभी 3 1 हो जाते हैं, और 4 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!
वास्तव में इतना कोड नहीं है। ऊपर दिए गए अधिकांश कोड टिप्पणी हैं। यहां ठीक उसी कोड है, टिप्पणियों के बिना।
@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!
यह समाधान विंडोज बैच के भीतर यूनिक्स शैली के तर्क प्रदान करता है। यह विंडोज के लिए आदर्श नहीं है - बैच में आमतौर पर आवश्यक तर्कों से पहले के विकल्प होते हैं और विकल्पों के साथ उपसर्ग होता है /
।
इस समाधान में उपयोग की जाने वाली तकनीकों को आसानी से विंडोज शैली के विकल्पों के लिए अनुकूलित किया जाता है।
- पार्सिंग लूप हमेशा एक विकल्प की तलाश में रहता है
%1
, और यह तब तक जारी रहता है जब तक कि आर्ग 1 के साथ शुरू नहीं होता है/
- ध्यान दें कि यदि नाम से शुरू होता है, तो SET असाइनमेंट को उद्धरण के भीतर संलग्न किया जाना चाहिए
/
।
SET /VAR=VALUE
विफल रहता है
SET "/VAR=VALUE"
काम करता है। मैं पहले से ही अपने समाधान में यह कर रहा हूँ।
- मानक Windows शैली पहले आवश्यक तर्क मान के साथ शुरू होने की संभावना को छोड़ देती है
/
। इस सीमा को स्पष्ट रूप से परिभाषित //
विकल्प को नियोजित करके समाप्त किया जा सकता है जो विकल्प पार्सिंग लूप से बाहर निकलने के लिए एक संकेत के रूप में कार्य करता है। //
"विकल्प" के लिए कुछ भी संग्रहीत नहीं किया जाएगा ।
2015-12-28 अपडेट करें:!
विकल्प मानों के लिए समर्थन
ऊपर दिए गए कोड में, प्रत्येक तर्क का विस्तार किया जाता है जबकि विलंबित विस्तार को सक्षम किया जाता है, जिसका अर्थ है कि !
सबसे अधिक संभावना छीन ली गई है, या जैसे कुछ !var!
का विस्तार किया गया है। इसके अलावा, ^
अगर !
मौजूद है तो भी छीन लिया जा सकता है। निम्न-टिप्पणी वाले कोड में निम्नलिखित छोटा संशोधन सीमा को हटा देता है !
और ^
विकल्प मानों में संरक्षित हैं।
@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!