2016-02-06 2 views
6

Sto cercando di sostituire tutte le istanze di con {~} in uno script batch in modo che VBScript non presuma che intendo colpire invio quando lo passo a sendKeys, tuttavia, la variabile continua a essere sostituita dalla stringa letterale destinata a sostituiscilo.Come uscire da "~" nel comando set/replace?

Ad esempio,

rem Intended function; 
rem input = Hello~world. 
rem output = Hello{~}world. 

Tuttavia ogni volta, neanche fa nulla, o imposta refVar alla stringa letterale refVar:~={~}.

Ecco cosa ho provato;

set refVar=!refVar:~={~}! 

rem Also 
set refVar=%refVar:~={~}% 

rem and... 
set refVar=%refVar:^~=^{^~^}% 

rem and of course; 
set refVar=!refVar:^~=^{^~^}! 

rem and even: 

set "pre=~" 
set "post={~}" 
set refVar=%refVar:!pre!=!post!% 

rem and the same set commands with; 
set refVar=!refVar:%pre%=%post%! 

Mi manca un modo per sfuggire a questo? Sono abbastanza sicuro che abbia a che fare con ~ come carattere posizionale in un comando simile.

Capisco che c'è un modo per risolvere questo problema in VBScript, ma questo mi infastidisce per avere una restrizione senza apparenti soluzioni alternative per me.

risposta

2

I offrire una soluzione forse più semplice per sostituire ogni carattere di tilde nella stringa di valore di una variabile di ambiente con una tilde in parentesi graffe rispetto a Dennis van Gils che funziona bene anche per questa attività.

@echo off 
set "TestVar=hello~world." 
echo TestVar has value: %TestVar% 
call :SpecialReplace "TestVar" 
echo TestVar has value: %TestVar% 
goto :EOF 


rem The subroutine below expects the name of an environment variable as 
rem parameter. The subroutine does nothing if called without parameter. 
rem Also nothing is done if specified environment variable is not defined. 

rem Each tilde character in value of this environment variable is replaced 
rem by {~} by this subroutine. 

rem Note: This subroutine can be also easily modified to replace other 
rem special characters like the equal sign by a different string which 
rem can be also no string in case of special character should be removed. 
rem Just modify Search and Replace variables for a different replace. But 
rem be aware of more code must be changed if search string is longer than 
rem one character. Length of replace string does not matter on code below. 

:SpecialReplace 
if "%~1" == "" exit /B 
setlocal EnableDelayedExpansion 
set "VarName=%~1" 
set "VarValue=!%VarName%!" 
if "!VarValue!" == "" endlocal & exit /B 
set "NewValue=" 
set "Search=~" 
set "Replace={~}" 

:ReplaceLoop 
if "!VarValue:~0,1!" == "!Search!" (
    set "NewValue=!NewValue!!Replace!" 
) else (
    set "NewValue=!NewValue!!VarValue:~0,1!" 
) 
set "VarValue=!VarValue:~1!" 
if not "!VarValue!" == "" goto ReplaceLoop 
endlocal & set "%VarName%=%NewValue%" & exit /B 

per la comprensione dei comandi utilizzati e come funzionano, aprire una finestra del prompt dei comandi, eseguire i seguenti comandi c'è, e leggere interamente tutte le pagine di aiuto visualizzati per ogni comando con molta attenzione.

  • call /?
  • echo /?
  • endlocal /?
  • exit /?
  • goto /?
  • if /?
  • rem /?
  • set /?
  • setlocal /?

E vedi rispondere anche Single line with multiple commands using Windows batch file per capire come le due linee con più comandi con un singolo e commerciale tra i comandi funzionano.


Una variante di cui sopra con una subroutine estesa SpecialReplace supporto ricerca ora variabile e sostituzione di stringhe indicato come secondo e terzo parametro come suggerito da Dennis van Gils.

@echo off 
set "TestVar=hello~world." 
echo TestVar has value: %TestVar% 
call :SpecialReplace "TestVar" "~" "{~}" 
echo TestVar has value: %TestVar% 
goto :EOF 


rem The subroutine below expects the name of an environment variable as 
rem first parameter. The subroutine does nothing if called without parameter. 
rem Also nothing is done if specified environment variable is not defined. 

rem The second parameter must be the search string. The subroutine does 
rem nothing if there is no second parameter which can be also just one 
rem or two double quotes enclosed in double quotes. 

rem The third parameter is an optional replace string. Without a third 
rem parameter all occurrences of search string are removed from value 
rem of the specified environment variable. 

rem With just one or two double quotes in double quotes as search string 
rem no replace string can be defined because command processor does not 
rem parse the quote strings as most would expect it. But """ or """" as 
rem search string with no replace string works and results in removing 
rem all double quotes or all occurrences of two double quotes. 

:SpecialReplace 
if "%~1" == "" exit /B 
setlocal EnableDelayedExpansion 
set "VarName=%~1" 
set "VarValue=!%VarName%!" 
if "!VarValue!" == "" endlocal & exit /B 
set "NewValue=" 
set "Search=%~2" 
if "!Search!" == "" endlocal & exit /B 
set "Replace=%~3" 
set "Length=0" 

:GetLength 
if "!Search:~%Length%,1!" == "" goto ReplaceLoop 
set /A Length+=1 
goto GetLength 

:ReplaceLoop 
if "!VarValue:~0,%Length%!" == "!Search!" (
    set "NewValue=!NewValue!!Replace!" 
    set "VarValue=!VarValue:~%Length%!" 
) else (
    set "NewValue=!NewValue!!VarValue:~0,1!" 
    set "VarValue=!VarValue:~1!" 
) 
if not "!VarValue!" == "" goto ReplaceLoop 
endlocal & set "%VarName%=%NewValue%" & exit /B 
+0

Questa è davvero una risposta chiara. Potrebbe essere meglio usare gli argomenti per la ricerca e sostituire la stringa, per una più facile usabilità. –

+0

Haha, conosco i comandi di scripting di file batch per la maggior parte, ma suppongo che i comandi di aiuto siano per gli altri che cercano la stessa risposta. In ogni caso, grazie, felice di sapere che la soluzione non era solo un certo ordine di delimitatori che non ero riuscito a provare. – Bloodied

+0

@Arescet Sì, ho aggiunto l'elenco dei comandi usati e come ottenere aiuto su di essi per chiunque stia leggendo questa risposta, come faccio quasi sempre sulle risposte relative ai file batch. Era chiaro per me dalla tua domanda che tu conoscessi tutti quei comandi molto bene. – Mofi

4

Così ho fatto un po 'di scavo e sembra che sia davvero, davvero difficile. Tuttavia, ho trovato this, che risolve il problema, dando noi questo come codice:

@echo off 

set "input=Hello~world." 
echo input: %input% 

call :wavereplacer "%input%" {tilde} output 
set output=%output:{tilde}={~}% 
echo output: %output% 
pause 

::your own code should be above this. 
goto :eof 

:wavereplacer String Replacer [RtnVar] 
setlocal 
rem the result of the operation will be stored here 
set "result=#%~1#" 
set "replacer=%~2" 
call :strlen0.3 result wl 
call :strlen0.3 replacer rl 

:start 

    set "part1=" 
    set "part2=" 

    rem splitting the string on two parts 
    for /f "tokens=1* delims=~" %%w in ("%result%") do (
    set "part1=%%w" 
    set "part2=%%x" 
) 

    rem calculating the count replace strings we should use 
    call :strlen0.3 part1 p1l 
    call :strlen0.3 part2 p2l 
    set /a iteration_end=wl-p1l-p2l 

    rem creating a sequence with replaced strings 
    setlocal enableDelayedExpansion 
    set "sequence=" 
    for /l %%i in (1,1,%iteration_end%) do (
    set sequence=!sequence!%replacer% 
) 
    endlocal & set "sequence=%sequence%" 

    rem adjust the string length 
    set /a wl=wl+iteration_end*(rl-1) 

    rem replacing for the current iteration 
    set result=%part1%%sequence%%part2% 
    rem if the second part is empty the task is over 
    if "%part2%" equ "" (
    set result=%result:~1,-1% 
    goto :endloop 
) 


    goto :start 

:endloop 
endlocal & if "%~3" neq "" (set %~3=%result%) else echo %result% 
exit /b 

:strlen0.3 StrVar [RtnVar] 
    setlocal EnableDelayedExpansion 
    set "s=#!%~1!" 
    set "len=0" 
    for %%A in (2187 729 243 81 27 9 3 1) do (
    set /A mod=2*%%A 
    for %%Z in (!mod!) do (
     if "!s:~%%Z,1!" neq "" (
     set /a "len+=%%Z" 
     set "s=!s:~%%Z!" 

    ) else (
     if "!s:~%%A,1!" neq "" (
      set /a "len+=%%A" 
      set "s=!s:~%%A!" 
     ) 
    ) 
    ) 
) 
    endlocal & if "%~2" neq "" (set %~2=%len%) else echo %len% 
exit /b 

Questo non può sostituire ~ a {~} immediatamente, è necessario utilizzare una sostituzione temporanea, in questo caso {tilde}

1
@ECHO Off 
SETLOCAL ENABLEDELAYEDEXPANSION 
SET "input=hello~world" 

SET "xhead=!input:%input:*~=%=!" 
SET "output=!input:%xhead%=%xhead:~0,-1%{~}!" 

ECHO %input% --^> %output% 

GOTO :EOF 

Questo sembra funzionare condizione che vi sia esattamente un ~ e che non è il carattere iniziale o terminale.